Запаздывает с использованием холста во время прокрутки
Я пытаюсь создать контур / обводку текста в Android. Поскольку не существует "простого способа" создания обводки, я пытаюсь реализовать это с помощью canvas. Я попробовал этот класс расширения Textview (OutlineTextView.java):
import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Paint;
import android.graphics.Typeface;
import android.text.Layout;
import android.text.StaticLayout;
import android.text.TextPaint;
import android.util.AttributeSet;
import android.widget.TextView;
public class OutlineTextView extends TextView {
public OutlineTextView(Context context) {
super(context);
initPaint();
}
public OutlineTextView(Context context, AttributeSet attrs) {
super(context, attrs);
initPaint();
}
public OutlineTextView(Context context, AttributeSet attrs, int defStyle) {
super(context, attrs, defStyle);
initPaint();
}
private void initPaint() {
mTextPaint = new TextPaint();
mTextPaint.setAntiAlias(true);
mTextPaint.setTextSize(getTextSize());
mTextPaint.setColor(mColor);
mTextPaint.setStyle(Paint.Style.FILL);
mTextPaint.setTypeface(getTypeface());
mTextPaintOutline = new TextPaint();
mTextPaintOutline.setAntiAlias(true);
mTextPaintOutline.setTextSize(getTextSize());
mTextPaintOutline.setColor(mBorderColor);
mTextPaintOutline.setStyle(Paint.Style.STROKE);
mTextPaintOutline.setTypeface(getTypeface());
mTextPaintOutline.setStrokeWidth(mBorderSize);
}
public void setText(String text) {
super.setText(text);
mText = text.toString();
requestLayout();
invalidate();
}
public void setTextSize(float size) {
super.setTextSize(size);
requestLayout();
invalidate();
initPaint();
}
public void setTextColor(int color) {
super.setTextColor(color);
mColor = color;
invalidate();
initPaint();
}
public void setShadowLayer(float radius, float dx, float dy, int color) {
super.setShadowLayer(radius, dx, dy, color);
mBorderSize = radius;
mBorderColor = color;
requestLayout();
invalidate();
initPaint();
}
public void setTypeface(Typeface tf, int style) {
super.setTypeface(tf, style);
requestLayout();
invalidate();
initPaint();
}
public void setTypeface(Typeface tf) {
super.setTypeface(tf);
requestLayout();
invalidate();
initPaint();
}
@Override
protected void onDraw(Canvas canvas) {
Layout layout = new StaticLayout(getText(), mTextPaintOutline, getWidth(), Layout.Alignment.ALIGN_CENTER, mSpacingMult, mSpacingAdd, mIncludePad);
layout.draw(canvas);
layout = new StaticLayout(getText(), mTextPaint, getWidth(), Layout.Alignment.ALIGN_CENTER, mSpacingMult, mSpacingAdd, mIncludePad);
layout.draw(canvas);
}
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
Layout layout = new StaticLayout(getText(), mTextPaintOutline, measureWidth(widthMeasureSpec), Layout.Alignment.ALIGN_CENTER, mSpacingMult, mSpacingAdd, mIncludePad);
int ex = (int) (mBorderSize * 2 + 1);
setMeasuredDimension(measureWidth(widthMeasureSpec) + ex, measureHeight(heightMeasureSpec) * layout.getLineCount() + ex);
}
private int measureWidth(int measureSpec) {
int result = 0;
int specMode = MeasureSpec.getMode(measureSpec);
int specSize = MeasureSpec.getSize(measureSpec);
if (specMode == MeasureSpec.EXACTLY) {
result = specSize;
} else {
result = (int) mTextPaintOutline.measureText(mText) + getPaddingLeft() + getPaddingRight();
if (specMode == MeasureSpec.AT_MOST) {
result = Math.min(result, specSize);
}
}
return result;
}
private int measureHeight(int measureSpec) {
int result = 0;
int specMode = MeasureSpec.getMode(measureSpec);
int specSize = MeasureSpec.getSize(measureSpec);
mAscent = (int) mTextPaintOutline.ascent();
if (specMode == MeasureSpec.EXACTLY) {
result = specSize;
} else {
result = (int) (-mAscent + mTextPaintOutline.descent()) + getPaddingTop() + getPaddingBottom();
if (specMode == MeasureSpec.AT_MOST) {
result = Math.min(result, specSize);
}
}
return result;
}
private TextPaint mTextPaint;
private TextPaint mTextPaintOutline;
private String mText = "";
private int mAscent = 0;
private float mBorderSize;
private int mBorderColor;
private int mColor;
private float mSpacingMult = 1.0f;
private float mSpacingAdd = 0;
private boolean mIncludePad = true;
}
Я использую этот класс в своей основной деятельности следующим образом:
OutlineTextView t2b = new OutlineTextView(this);
t2b.setGravity(Gravity.CENTER | Gravity.CENTER_HORIZONTAL);
t2b.setTextSize(44);
t2b.setTextColor(Color.parseColor(Color.Green));
t2b.setShadowLayer(10, 1, 1, Color.BLACK);
Typeface type2 = Typeface.createFromAsset(getAssets(),"fonts/HelveticaLTStd-Bold.otf");
t2b.setTypeface(type2);
t2b.setText(percentage + "%");
fl.addView(t2b);
fl это FrameLayout. Вывод теперь - штриховой текст, и он выглядит именно так, как я хочу. My View - это длинный список плиток, на каждом из которых есть штриховой текст. Текст не моя проблема. Проблема в производительности при прокрутке. Если я прокручиваю вниз или вверх, производительность замедляется и прокрутка происходит не плавно. Я думаю, что проблема в том, как я рисую штриховые линии. Я пробовал много вариантов текста обводки, а также пробовал все предложенные варианты линий обводки отсюда. Каждый раз у меня одна и та же проблема. Если я рисую текст без линий обводки (но все еще использую OutlineTextView), он снова прокручивается плавно, без задержек и замедлений производительности. Кажется, что рисование линий обводки замедляет работу системы.
PS: Этот OutlineTextView.java не мой, я только что получил его из Интернета, так как кажется, что он нарисовал штриховой текст.
Буду благодарен, если кто-нибудь сможет мне помочь.