Android ClickableSpan n'appelle pas onClick

150

Je crée un ClickableSpan, et il s'affiche correctement avec le texte approprié souligné. Cependant, les clics ne sont pas enregistrés. Savez-vous ce que je fais mal ???

Merci, Victor

Voici l'extrait de code:

view.setText("This is a test");
ClickableSpan span = new ClickableSpan() {
    @Override
    public void onClick(View widget) {
        log("Clicked");
    }
};
view.getText().setSpan(span, 0, view.getText().length(), Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
Victor Grazi
la source

Réponses:

430

Avez-vous essayé de définir MovementMethod sur le TextView qui contient la plage? Vous devez faire cela pour que le clic fonctionne ...

tv.setMovementMethod(LinkMovementMethod.getInstance());
Marc Attinasi
la source
Ne fonctionne pas bien si tvest de type EditText, c'est vrai que vous pouvez cliquer sur la plage mais pas l'éditer normalement.
FIG-GHD742
Merci beaucoup! C'est du travail pour moi aussi! pouvez-vous m'expliquer pourquoi sur ce paramètre?
alfo888_ibg
63
Bien sûr, j'ai besoin de définir ce que la documentation appelle un "gestionnaire de touches fléchées" pour faire fonctionner un gestionnaire de clics. Tellement évident! (╯ ° □ °) ╯︵ ┻━┻
adamdport
Cela fonctionne, mais je ne saurai jamais vraiment pourquoi ce n'est pas le comportement par défaut.
EpicPandaForce
Et Google a oublié de mentionner qu'appeler setMovementMethod fait que le "ellipsize" ne fonctionne pas ... Il semble donc que la bonne approche consiste à implémenter manuellement un TouchListener et à partir de là ...
slott
4

Après quelques essais et erreurs, la séquence de réglage tv.setMovementMethod(LinkMovementMethod.getInstance());importe.

Voici mon code complet

String stringTerms = getString(R.string.sign_up_terms);
Spannable spannable = new SpannableString(stringTerms);
int indexTermsStart = stringTerms.indexOf("Terms");
int indexTermsEnd = indexTermsStart + 18;
spannable.setSpan(new UnderlineSpan(), indexTermsStart, indexTermsEnd, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
spannable.setSpan(new ForegroundColorSpan(getColor(R.color.theme)), indexTermsStart, indexTermsEnd, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
spannable.setSpan(new ClickableSpan() {
    @Override
    public void onClick(View widget) {
        Log.d(TAG, "TODO onClick.. Terms and Condition");
    }
}, indexTermsStart, indexTermsEnd, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);

int indexPolicyStart = stringTerms.indexOf("Privacy");
int indexPolicyEnd = indexPolicyStart + 14;
spannable.setSpan(new UnderlineSpan(), indexPolicyStart, indexPolicyEnd, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
spannable.setSpan(new ForegroundColorSpan(getColor(R.color.theme)), indexPolicyStart, indexPolicyEnd, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
spannable.setSpan(new ClickableSpan() {
    @Override
    public void onClick(View widget) {
        Log.d(TAG, "TODO onClick.. Privacy Policy");
    }
}, indexPolicyStart, indexPolicyEnd, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);

TextView textViewTerms = (TextView) findViewById(R.id.sign_up_terms_text);
textViewTerms.setText(spannable);
textViewTerms.setClickable(true);
textViewTerms.setMovementMethod(LinkMovementMethod.getInstance());
Twelvester
la source
4

Fonction Kotlin util:

fun setClickable(textView: TextView, subString: String, handler: () -> Unit, drawUnderline: Boolean = false) {
    val text = textView.text
    val start = text.indexOf(subString)
    val end = start + subString.length

    val span = SpannableString(text)
    span.setSpan(ClickHandler(handler, drawUnderline), start, end, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE)

    textView.linksClickable = true
    textView.isClickable = true
    textView.movementMethod = LinkMovementMethod.getInstance()

    textView.text = span
}

class ClickHandler(
        private val handler: () -> Unit,
        private val drawUnderline: Boolean
) : ClickableSpan() {
    override fun onClick(widget: View?) {
        handler()
    }

    override fun updateDrawState(ds: TextPaint?) {
        if (drawUnderline) {
            super.updateDrawState(ds)
        } else {
            ds?.isUnderlineText = false
        }
    }
}

Usage:

Utils.setClickable(textView, subString, {handleClick()})
Onze
la source
1

Approche directe à Kotlin

  val  textHeadingSpannable = SpannableString(resources.getString(R.string.travel_agent))


           val clickSpan = object : ClickableSpan(){
               override fun onClick(widget: View) {

                // Handel your click
               }
           }
            textHeadingSpannable.setSpan(clickSpan,104,136,Spannable.SPAN_INCLUSIVE_EXCLUSIVE)

            tv_contact_us_inquire_travel_agent.movementMethod = LinkMovementMethod.getInstance()
            tv_contact_us_inquire_travel_agent.text = textHeadingSpannable
mughil
la source