On Android, you can use ClickableSpan to create a section of text that can be clicked on
tv = (TextView) findViewById(R.id.tv_tsm_test); SpannableStringBuilder Builder = new SpannableStringBuilder(" this is a connection "); builder.setSpan(new ClickableSpan() { @Override public void onClick(@NonNull View widget) { tv.setBackgroundColor(Color.GREEN); } }, 3, 5, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); tv.setText(builder); tv.setMovementMethod(LinkMovementMethod.getInstance()); tv.setAutoLinkMask(Linkify.WEB_URLS);Copy the code
The result is as follows
Click connect to call back the ClickableSpan onClick method to change the background to green, a very simple application. Now the product has asked us to add a long press event to the TextView, thinking it would be as simple as a few lines of code
tv = (TextView) findViewById(R.id.tv_tsm_test); SpannableStringBuilder Builder = new SpannableStringBuilder(" this is a connection "); builder.setSpan(new ClickableSpan() { @Override public void onClick(@NonNull View widget) { tv.setBackgroundColor(Color.GREEN); } }, 3, 5, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); tv.setText(builder); tv.setMovementMethod(LinkMovementMethod.getInstance()); tv.setAutoLinkMask(Linkify.WEB_URLS); tv.setOnLongClickListener(new View.OnLongClickListener() { @Override public boolean onLongClick(View v) { tv.setBackgroundColor(Color.RED); return true; }});Copy the code
The ClickableSpan onClick event is also called back to the ClickableSpan onClick event. This is caused by LinkMovementMethod
The ClickableSpan onClick event is also called back to the ClickableSpan onClick event. This is caused by LinkMovementMethod
@Override public boolean onTouchEvent(TextView widget, Spannable buffer, MotionEvent event) { int action = event.getAction(); 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]; ACTION_UP) {if (link instanceof TextLinkSpan) {((TextLinkSpan)); link).onClick( widget, TextLinkSpan.INVOCATION_METHOD_TOUCH); } else { link.onClick(widget); } } else if (action == MotionEvent.ACTION_DOWN) { ........ } return true; } else { Selection.removeSelection(buffer); } } return super.onTouchEvent(widget, buffer, event); }Copy the code
We need to change the method so that when the event is lifted, it will respond to the ClickableSpan onClick event regardless of whether the event is clicked or long pressed. We need to change the method so that the event will not respond to the ClickableSpan onClick event when the event is extended
private static final long CLICK_DELAY = 1*1000; if (link.length ! = 0) { switch (action){ case MotionEvent.ACTION_UP: long flag=(System.currentTimeMillis() - lastClickTime); if (flag< CLICK_DELAY) { link[0].onClick(widget); } return true; case MotionEvent.ACTION_DOWN: Selection.setSelection(buffer, buffer.getSpanStart(link[0]), buffer.getSpanEnd(link[0])); lastClickTime = System.currentTimeMillis(); return true; } } else { Selection.removeSelection(buffer); }Copy the code
The modified code now looks like this. This time, we will try again and find that the onClick event of ClickableSpan does not ring during the long press. Do you think this is over She wants to copy all the text freely. The long press copy experience is not good. She doesn’t want to copy
android:textIsSelectable="true"
Copy the code
This property is set to true, but when tested, ClickableSpan is called twice, what? I have just seen it in the onTouch of LinkMovementMethod. There is only one click event, so why did it respond twice? After a lot of trouble, TextView onTouchEvent also has the corresponding judgment
/ / / lift operation, focus and have the final Boolean touchIsFinished = (action = = MotionEvent. ACTION_UP) && (mEditor = = null | |! mEditor.mIgnoreActionUpEvent) && isFocused(); //enable and text is Spannable if ((mMovement! = null || onCheckIsTextEditor()) && isEnabled() && mText instanceof Spannable && mLayout ! = null) { boolean handled = false; if (mMovement ! = null) { handled |= mMovement.onTouchEvent(this, mSpannable, event); } final boolean textIsSelectable = isTextSelectable(); /// ClickableSpan is called when all conditions are met and the link can be clicked and setAutoLinkMask is set and textIsSelectable=true If (touchIsFinished && mLinksClickable && mAutoLinkMask! = 0 && textIsSelectable) { // The LinkMovementMethod which should handle taps on links has not been installed // on non editable text that support text selection. // We reproduce its behavior here to open links for these. ClickableSpan[] links = mSpannable.getSpans(getSelectionStart(), getSelectionEnd(), ClickableSpan.class); if (links.length > 0) { links[0].onClick(this); handled = true; }}Copy the code
In the actual project, the original setAutoLinkMask(linkify.web_urls) cannot be changed because the code is hereditary and important attributes cannot be modified. This attribute is removed, so can only modify the mLinksClickable, let him not satisfy, will not affect our event, in the actual development process can also be setMovementMethod (LinkMovementMethod. GetInstance ()); This code removes, or tries to remove, setAutoLinkMask(Linkify.WEB_URLS); So this code, I’m going to pick
android:linksClickable="false"
Copy the code
This attribute is added in the layout, can achieve the result that we want, the click event will respond only once, but in actual development process, I modified code is inside the component, using the application of several components, then according to the function switch in the code to modify the properties of dynamic code for the actual project
SpannableStringBuilder builder = SpannableUtil.addInnerLink(vh.tv, msg, color, needLinkUnderLine, new SpannableUtil.LinkCallback() { @Override public void onLinkClick(String originText, String link, int startIndex, int endIndex) { } }); vh.tv.setText(SpannableUtil.addPhoneLink(msg.getMsgContent(), builder, mExtAdapter.isAddPhoneLink(), color, new SpannableUtil.LinkCallback() { @Override public void onLinkClick(String originText, String link, int startIndex, int endIndex) { } }), TextView.BufferType.SPANNABLE); If (open free replication){vh.tv.setLinksclickable (false); vh.tv.setTextIsSelectable(true); }Copy the code
SetMovementMethod (Selectable? ArrowKeyMovementMethod.getInstance() : null); In this case, if set to copy freely, then use ArrowKeyMovementMethod, otherwise set MovementMethod to null, replacing our LinkMovementMethod, so the vh.tv.setTextIsSelectable(true); This method is advanced to the ancestral setMovementMethod code, and the problem is resolved.
Author: Tian Shouming, Research and development Center of Freely Large front end
Recruitment information
Free big front end R&D center recruiting new students!
FE/IOS/Android engineers look over
Company benefits include:
- Five insurances and one housing fund in full, plus commercial insurance
- Free gym + annual physical examination
- 10 percent discount for rent near the company
- 2 promotions per year
Office location: Beijing Jiuxianqiao Putian Industrial Science and Technology Park welcome you to join us! Please send your resume to [email protected], or add wechat V-nice-V for details!