在TextView中动态改变文本颜色的最高效方法

5

我希望能够使用计时器多次更改文本的部分颜色,这与IT技术有关。最简单的方法是:

SpannableStringBuilder ssb = new SpannableStringBuilder(mainText);
ForegroundColorSpan span = new ForegroundColorSpan(Color.BLUE);
ssb.setSpan(span, start, end, 0);
tv.setText(ssb);

但如果我在一秒钟内多次运行上面的代码,实际上每次都会改变TextView的整个(大)文本,因此会在低端设备上产生不必要的内存-CPU负载。

我如何在TextView上有一个单独的Span,并且只更改Span的起始和结束位置?

这样做是否有效,或者幕后会发生完整的文本替换?

我的文本是固定的,永远不会改变。

3个回答

8
解决不调用setText方法时的跨度移动问题的方案:
    final TextView tv = new TextView(this);
    tv.setTextSize(32);
    setContentView(tv);

    SpannableStringBuilder ssb = new SpannableStringBuilder("0123456789012345678901234567890123456789");
    ssb.append(ssb).append(ssb);
    tv.setText(ssb, BufferType.SPANNABLE);
    final Spannable sp = (Spannable) tv.getText();
    final ForegroundColorSpan span = new ForegroundColorSpan(0xffff0000);
    Runnable action = new Runnable() {
        @Override
        public void run() {
            sp.setSpan(span, start, start + 4, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
            start++;
            if (start <= sp.length() - 4) {
                tv.postDelayed(this, 50);
            }
        }
    };
    tv.postDelayed(action, 1000);

动态颜色更改解决方案:

class HSVSpan extends CharacterStyle {
    int color;
    float[] hsv = {0, 1, 1};

    @Override
    public void updateDrawState(TextPaint tp) {
        tp.setColor(color);
    }

    public void update() {
        hsv[0] += 5;
        hsv[0] %= 360;
        color = Color.HSVToColor(hsv);
//        Log.d(TAG, "update " + Integer.toHexString(color));
    }
}

以及测试代码:

    final TextView tv = new TextView(this);
    setContentView(tv);
    SpannableStringBuilder ssb = new SpannableStringBuilder("0123456789");
    final HSVSpan span = new HSVSpan();
    ssb.setSpan(span, 2, 6, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
    tv.setText(ssb);
    tv.setTextSize(32);

    Runnable action = new Runnable() {
        @Override
        public void run() {
            span.update();
            tv.invalidate();
            tv.postDelayed(this, 50);
        }
    };
    action.run();

你有这方面的样例吗?CharacterStyle是ForegroundColorSpan的超类...那Editable呢,它会在更改时更新TextView吗? - Mohsen Afshin
谢谢你的代码,但我认为你没有理解问题的要点。我有一个固定的颜色和动态的起始和结束跨度位置。跨度位置仅在每次运行时更改,而不是跨度的样式。 - Mohsen Afshin
好的,扩展ReplacementSpan并为每个TextView行创建一个实例,现在您可以随心所欲地进行绘制。 - pskink
利用底层SpanWatcher和TextView的其余内部机制,实现最高效的解决方案。谢谢。 - Mohsen Afshin
SpanWatcher?它如何帮助?SpanWatcher只通知有关跨度的添加、删除和更改。 - pskink
所发生的是一个 span 改变,这里没有问题。结果非常好。 :-) - Mohsen Afshin

1
执行以下代码一次:
SpannableStringBuilder ssb = new SpannableStringBuilder(mainText);
ForegroundColorSpan span = new ForegroundColorSpan(Color.BLUE);

每当你想要更改span时,请执行以下操作:

ssb.clearSpans()
ssb.setSpan(span, start, end, 0);
tv.setText(ssb);

@vipual,我已经完成了,现在正在寻找比设置整个文本更好的方法。 - Mohsen Afshin
这样做会导致性能不佳吗?当更改发生时,您需要一些方法来刷新用户界面。 - vipul mittal
其实这些都不是坏的,我想尽可能地减少CPU内存负载。我知道刷新是必要的,但实际上每次运行只有一些字符会改变。 - Mohsen Afshin
@MohsenAfshin 实际上它们都没有改变:只有 span 改变了:请查看我的更新答案。 - pskink

0

今天下午我遇到了相同的问题,这是我的解决方法:

tv.setText(yourText, TextView.BufferType.SPANNABLE);
ForegroundColorSpan span = new ForegroundColorSpan(Color.BLUE);
((Spannable) article.getText()).setSpan(span, start, end, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);

我发现这是一种更有效的方法,在我的项目中,使用set-whole-text方法需要400~600毫秒,而用这种方法只需要0或1毫秒。


1
比什么更高效? - Suragch

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接