Combining Spannable with String.format()

前端 未结 5 468
刺人心
刺人心 2021-01-31 09:28

Suppose you have the following string:

String s = \"The cold hand reaches for the %1$s %2$s Ellesse\'s\";
String old = \"old\"; 
String tan = \"tan\"; 
String fo         


        
相关标签:
5条回答
  • 2021-01-31 09:45

    I had the same issue pop up, but found a really great solution here: Android: How to combine Spannable.setSpan with String.format?

    Check out the solution offered by @george-steel. He created a custom version of String.format which preserves spans.

    Example use:

    Spanned toDisplay = SpanFormatter.format(getText(R.string.foo), bar, baz, quux);
    
    0 讨论(0)
  • 2021-01-31 09:49

    I have created a version of String.format that works with spannables. Download it and use it just like the normal version. In your case you would put the spans around the format specifiers (possibly using strings.xml). In the output, they would be around whatever those specifiers were replaced with.

    0 讨论(0)
  • 2021-01-31 09:49

    Using Spannables like that is a headache -- this is probably the most straightforward way around:

    String s = "The cold hand reaches for the %1$s %2$s Ellesse's";
    String old = "<font color=\"blue\">old</font>"; 
    String tan = "<strike>tan</strike>"; 
    String formatted = String.format(s,old,tan); //The cold hand reaches for the <font color="blue">old</font> <strike>tan</strike> Ellesse's
    
    Spannable spannable = Html.fromHtml(formatted);
    

    Problem: this does not put in a StrikethroughSpan. To make the StrikethroughSpan, we borrow a custom TagHandler from this question.

    Spannable spannable = Html.fromHtml(text,null,new MyHtmlTagHandler());
    

    MyTagHandler:

    public class MyHtmlTagHandler implements Html.TagHandler {
        public void handleTag(boolean opening, String tag, Editable output,
                              XMLReader xmlReader) {
            if (tag.equalsIgnoreCase("strike") || tag.equals("s")) {
                processStrike(opening, output);
            }
        }
        private void processStrike(boolean opening, Editable output) {
            int len = output.length();
            if (opening) {
                output.setSpan(new StrikethroughSpan(), len, len, Spannable.SPAN_MARK_MARK);
            } else {
                Object obj = getLast(output, StrikethroughSpan.class);
                int where = output.getSpanStart(obj);
                output.removeSpan(obj);
                if (where != len) {
                    output.setSpan(new StrikethroughSpan(), where, len, Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);
                }
            }
        }
    
        private Object getLast(Editable text, Class kind) {
            Object[] objs = text.getSpans(0, text.length(), kind);
            if (objs.length == 0) {
                return null;
            } else {
                for (int i = objs.length; i > 0; i--) {
                    if (text.getSpanFlags(objs[i - 1]) == Spannable.SPAN_MARK_MARK) {
                        return objs[i - 1];
                    }
                }
                return null;
            }
        }
    }
    
    0 讨论(0)
  • 2021-01-31 09:57

    I made simple Kotlin extension function that should solve String.format spanned issue:

    fun Context.getStringSpanned(@StringRes resId: Int, vararg formatArgs: Any): Spanned {
        var lastArgIndex = 0
        val spannableStringBuilder = SpannableStringBuilder(getString(resId, *formatArgs))
        for (arg in formatArgs) {
            val argString = arg.toString()
            lastArgIndex = spannableStringBuilder.indexOf(argString, lastArgIndex)
            if (lastArgIndex != -1) {
                (arg as? CharSequence)?.let {
                    spannableStringBuilder.replace(lastArgIndex, lastArgIndex + argString.length, it)
                }
                lastArgIndex += argString.length
            }
        }
    
        return spannableStringBuilder
    }
    
    0 讨论(0)
  • 2021-01-31 10:06

    I needed to replace %s placeholders in a String by a set of Spannables, and didn't find anything satisfying enough, so I implemented my own formatter as a kotlin String extension. Hopes it helps anyone.

    fun String.formatSpannable(vararg spans: CharSequence?): Spannable {
        val result = SpannableStringBuilder()
        when {
            spans.size != this.split("%s").size - 1 ->
                Log.e("formatSpannable",
                        "cannot format '$this' with ${spans.size} arguments")
            !this.contains("%s") -> result.append(this)
            else -> {
                var str = this
                var spanIndex = 0
                while (str.contains("%s")) {
                    val preStr = str.substring(0, str.indexOf("%s"))
                    result.append(preStr)
                    result.append(spans[spanIndex] ?: "")
                    str = str.substring(str.indexOf("%s") + 2)
                    spanIndex++
                }
                if (str.isNotEmpty()) {
                    result.append(str)
                }
            }
        }
        return result
    }
    

    and then usage as follows

    "hello %s kotlin %s world".formatSpannable(span0, span1)
    
    0 讨论(0)
提交回复
热议问题