我正在尝试使用ReplacementSpans来格式化EditText字段中的输入(而无需修改内容):
public class SpacerSpan extends ReplacementSpan {
@Override
public int getSize(Paint paint, CharSequence text, int start, int end, Paint.FontMetricsInt fm) {
return (int) paint.measureText(text.subSequence(start,end)+" ");
}
@Override
public void draw(Canvas canvas, CharSequence text, int start, int end, float x, int top, int y, int bottom, Paint paint) {
canvas.drawText(text.subSequence(start,end)+" ", 0, 2, x, y, paint);
}
}
这样做能按预期工作并在跨越的部分之后添加间距。但是,如果我同时应用ForegroundColorSpan,则所跨越部分的颜色不会被设置:
EditText edit = (EditText) findViewById(R.id.edit_text);
SpannableString content = new SpannableString("1234567890");
ForegroundColorSpan fontColor = new ForegroundColorSpan(Color.GREEN);
SpacerSpan spacer = new SpacerSpan();
content.setSpan(fontColor, 0, content.length(), Spanned.SPAN_INCLUSIVE_INCLUSIVE);
content.setSpan(spacer, 4, 5, Spanned.SPAN_MARK_MARK);
edit.setText(content);
结果看起来像 http://i.cubeupload.com/4Us5Zj.png 如果我应用一个AbsoluteSizeSpan,指定的字体大小也会应用于替换Span部分。这是预期行为吗?我有什么遗漏的吗?还是Android中的一个错误?
ReplacementSpan
的唯一具体实现是用图像替换文本,因此可能这些内容在用更多文本替换文本方面并没有得到很好的测试。ForegroundColorSpan
和AbsoluteSizeSpan
之间的一个区别是后者覆盖了updateMeasureState()
和updateDrawState()
,而ForegroundColorSpan
仅覆盖了updateDrawState()
。这是有道理的,因为前景色不会影响测量。但是,传递给draw()
的Paint
可能是错误的。 - CommonsWaredraw()
的Paint
不同(如果我记录设置的paint,则输出为):ForegroundColor. r: 0, g: 255, b: 0
Color during draw. r: 0, g: 0, b: 0
- Andreas WengerForegroundColorSpan
的子类,并覆盖updateMeasureState()
以执行updateDrawState()
所做的操作,然后尝试应用你的子类代替ForegroundColorSpan
。如果你的子类有效,那么好消息是你将确切知道问题的根源。坏消息是,如果Spanned
通过Bundle
传递,只有内置的ParcelableSpans
受支持,你可能会失去你的子类。 - CommonsWare