How to correctly draw text in an extended class for TextView?
I'm currently working on extending a TextView, adding an outline around the text. Thus far, the only problem I've been having is my inability to position the "outline" correctly behind a text. If I code the extended class like the one portrayed below, I get a label that looks like this:
Note: in the above screenshot, I set the fill color to white, and the stroke color to black.
What am I doing wrong?
public class OutlinedTextView extends TextView {
/* ===========================================================
* Constants
* =========================================================== */
private static final float OUTLINE_PROPORTION = 0.1f;
/* ===========================================================
* Members
* =========================================================== */
private final Paint mStrokePaint = new Paint();
private int mOutlineColor = Color.TRANSPARENT;
/* ===========================================================
* Constructors
* =========================================================== */
public OutlinedTextView(Context context) {
super(context);
this.setupPaint();
}
public OutlinedTextView(Context context, AttributeSet a开发者_运维知识库ttrs) {
super(context, attrs);
this.setupPaint();
this.setupAttributes(context, attrs);
}
public OutlinedTextView(Context context, AttributeSet attrs, int defStyle) {
super(context, attrs, defStyle);
this.setupPaint();
this.setupAttributes(context, attrs);
}
/* ===========================================================
* Overrides
* =========================================================== */
@Override
protected void onDraw(Canvas canvas) {
// Get the text to print
final float textSize = super.getTextSize();
final String text = super.getText().toString();
// setup stroke
mStrokePaint.setColor(mOutlineColor);
mStrokePaint.setStrokeWidth(textSize * OUTLINE_PROPORTION);
mStrokePaint.setTextSize(textSize);
mStrokePaint.setFlags(super.getPaintFlags());
mStrokePaint.setTypeface(super.getTypeface());
// Figure out the drawing coordinates
//mStrokePaint.getTextBounds(text, 0, text.length(), mTextBounds);
// draw everything
canvas.drawText(text,
super.getWidth() * 0.5f, super.getBottom() * 0.5f,
mStrokePaint);
super.onDraw(canvas);
}
/* ===========================================================
* Private/Protected Methods
* =========================================================== */
private final void setupPaint() {
mStrokePaint.setAntiAlias(true);
mStrokePaint.setStyle(Paint.Style.STROKE);
mStrokePaint.setTextAlign(Paint.Align.CENTER);
}
private final void setupAttributes(Context context, AttributeSet attrs) {
final TypedArray array = context.obtainStyledAttributes(attrs,
R.styleable.OutlinedTextView);
mOutlineColor = array.getColor(
R.styleable.OutlinedTextView_outlineColor, 0x00000000);
array.recycle();
// Force this text label to be centered
super.setGravity(Gravity.CENTER_HORIZONTAL);
}
}
Bah, that was stupid of me. I just needed to change-up that commented-out line:
super.getPaint().getTextBounds(text, 0, text.length(), mTextBounds);
In addition, for actually rendering the text, I need to average this view's height and the text's height:
// draw everything
canvas.drawText(text,
super.getWidth() * 0.5f, (super.getHeight() + mTextBounds.height()) * 0.5f,
mStrokePaint);
The entire code now reads as follows:
public class OutlinedTextView extends TextView {
/* ===========================================================
* Constants
* =========================================================== */
private static final float OUTLINE_PROPORTION = 0.1f;
/* ===========================================================
* Members
* =========================================================== */
private final Paint mStrokePaint = new Paint();
private final Rect mTextBounds = new Rect();
private int mOutlineColor = Color.TRANSPARENT;
/* ===========================================================
* Constructors
* =========================================================== */
public OutlinedTextView(Context context) {
super(context);
this.setupPaint();
}
public OutlinedTextView(Context context, AttributeSet attrs) {
super(context, attrs);
this.setupPaint();
this.setupAttributes(context, attrs);
}
public OutlinedTextView(Context context, AttributeSet attrs, int defStyle) {
super(context, attrs, defStyle);
this.setupPaint();
this.setupAttributes(context, attrs);
}
/* ===========================================================
* Overrides
* =========================================================== */
@Override
protected void onDraw(Canvas canvas) {
// Get the text to print
final float textSize = super.getTextSize();
final String text = super.getText().toString();
// setup stroke
mStrokePaint.setColor(mOutlineColor);
mStrokePaint.setStrokeWidth(textSize * OUTLINE_PROPORTION);
mStrokePaint.setTextSize(textSize);
mStrokePaint.setFlags(super.getPaintFlags());
mStrokePaint.setTypeface(super.getTypeface());
// Figure out the drawing coordinates
super.getPaint().getTextBounds(text, 0, text.length(), mTextBounds);
// draw everything
canvas.drawText(text,
super.getWidth() * 0.5f, (super.getHeight() + mTextBounds.height()) * 0.5f,
mStrokePaint);
super.onDraw(canvas);
}
/* ===========================================================
* Private/Protected Methods
* =========================================================== */
private final void setupPaint() {
mStrokePaint.setAntiAlias(true);
mStrokePaint.setStyle(Paint.Style.STROKE);
mStrokePaint.setTextAlign(Paint.Align.CENTER);
}
private final void setupAttributes(Context context, AttributeSet attrs) {
final TypedArray array = context.obtainStyledAttributes(attrs,
R.styleable.OutlinedTextView);
mOutlineColor = array.getColor(
R.styleable.OutlinedTextView_outlineColor, 0x00000000);
array.recycle();
// Force this text label to be centered
super.setGravity(Gravity.CENTER_HORIZONTAL);
}
}
I've been plugging away with some of these examples for a while, as none seemed to line up quite right, and once I finally got a handle on what is happening with text and put my maths hat on I changed my onDraw for this to be the following and it sits perfectly no matter the size of the text or what size and shape it's containing view is...
@Override
protected void onDraw(Canvas canvas) {
if (!isInEditMode()){
// Get the text to print
final float textSize = super.getTextSize();
final String text = super.getText().toString();
// setup stroke
mStrokePaint.setColor(mOutlineColor);
mStrokePaint.setStrokeWidth(textSize * mOutlineSize);
mStrokePaint.setTextSize(textSize);
mStrokePaint.setFlags(super.getPaintFlags());
mStrokePaint.setTypeface(super.getTypeface());
// draw everything
canvas.drawText(text,
(this.getWidth()-mStrokePaint.measureText(text))/2, this.getBaseline(),
mStrokePaint);
}
super.onDraw(canvas);
}
Turned out to be far less maths and Rect calculating than a lot of solutions used too.
Edit: Forgot to mention that I copy the textalign of the super in the initialisation and DON'T force it to center. The drawText position calculated here is always going to be the properly centered position for the stroked text.
I've been trying to make it work for some time and I have a solution, but it's for a special case only! It's possible to get Layout
object that is used inside the TextView
for drawing text. You can create a copy of this object and use it inside the onDraw(Canvas)
method.
final Layout originalLayout = super.getLayout();
final Layout layout = new StaticLayout(text, mStrokePaint,
originalLayout.getWidth(), originalLayout.getAlignment(),
originalLayout.getSpacingMultiplier(), originalLayout.getSpacingAdd(), true);
canvas.save();
canvas.translate( layout.getLineWidth(0) * 0.5f, 0.0f );
layout.draw(canvas);
canvas.restore();
But I'm sure that it's not a good way for drawing outlines. I don't know how to track changes in a TextView.getLayout()
object. Also it doesn't work for multiline TextView
s and different gravities. And eventually this code has very poor performance because it allocates a Layout
object on every draw. I don't understand exactly how it works, so I'd prefer not to use it.
There're a few attributes in the TextView
class, like android:shadowColor
, android:shadowDx
, android:shadowDy
and android:shadowRadius
. It's seems to me that they do the same thing you want to implement. So maybe you should try a simple TextView
at first.
精彩评论