Android Canvas.drawText

91

Mam widok, rysuję obiektem Canvas w metodzie onDraw (Canvas Canvas). Mój kod to:

Paint paint = new Paint();
paint.setColor(Color.WHITE);
paint.setStyle(Style.FILL);
canvas.drawPaint(paint);

paint.setColor(android.R.color.black);
paint.setTextSize(20);
canvas.drawText("Some Text", 10, 25, paint);

Problem polega na tym, że tekst nie jest widoczny w tle, co robię źle? Jeśli usunę canvas.drawPaint (paint) i paint.setColor (android.R.color.black) możesz zobaczyć tekst na ekranie .....

Gaz
źródło

Odpowiedzi:

153

Okazało się, że android.R.color.black to nie to samo, co Color.BLACK. Zmieniono kod na:

Paint paint = new Paint(); 
paint.setColor(Color.WHITE); 
paint.setStyle(Style.FILL); 
canvas.drawPaint(paint); 

paint.setColor(Color.BLACK); 
paint.setTextSize(20); 
canvas.drawText("Some Text", 10, 25, paint); 

i teraz wszystko działa dobrze !!

Gaz
źródło
35
Tak. Jeśli chcesz użyć definicji koloru w res/colors.xmlpliku z identyfikatorem R.color.black, nie możesz po prostu użyć identyfikatora. Jeśli chcesz uzyskać rzeczywistą wartość koloru z zasobów, użyjpaint.setColor(getResources().getColor(R.color.black));
Matt Gibson,
Ktoś wie jak wyciągnąć tekst w Android Canvas ShapeDrawable z RectShape ?
LOG_TAG
1
a do ustawienia rozmiaru tekstu w dpmożesz użyć tego
SMMousavi
Ważne jest, aby zresetować styl do FILL, w przeciwnym razie może obrysować tekst (potencjalnie grubymi liniami) i wyglądać naprawdę odważnie i brzydko.
Chase Roberts
Tutaj, w drawText ("Some Text", 10,25, paint); oznacza to, że lewy margines to 10, a górny to 25. mam rację?
Prince Dholakiya
37

Należy zauważyć, że dokumentacja zaleca używanie Layoutraczej a niż Canvas.drawTextbezpośrednio. Moja pełna odpowiedź na temat używania StaticLayoutjest tutaj , ale poniżej zamieszczę podsumowanie.

String text = "This is some text.";

TextPaint textPaint = new TextPaint();
textPaint.setAntiAlias(true);
textPaint.setTextSize(16 * getResources().getDisplayMetrics().density);
textPaint.setColor(0xFF000000);

int width = (int) textPaint.measureText(text);
StaticLayout staticLayout = new StaticLayout(text, textPaint, (int) width, Layout.Alignment.ALIGN_NORMAL, 1.0f, 0, false);
staticLayout.draw(canvas);

Oto pełniejszy przykład w kontekście widoku niestandardowego:

wprowadź opis obrazu tutaj

public class MyView extends View {

    String mText = "This is some text.";
    TextPaint mTextPaint;
    StaticLayout mStaticLayout;

    // use this constructor if creating MyView programmatically
    public MyView(Context context) {
        super(context);
        initLabelView();
    }

    // this constructor is used when created from xml
    public MyView(Context context, AttributeSet attrs) {
        super(context, attrs);
        initLabelView();
    }

    private void initLabelView() {
        mTextPaint = new TextPaint();
        mTextPaint.setAntiAlias(true);
        mTextPaint.setTextSize(16 * getResources().getDisplayMetrics().density);
        mTextPaint.setColor(0xFF000000);

        // default to a single line of text
        int width = (int) mTextPaint.measureText(mText);
        mStaticLayout = new StaticLayout(mText, mTextPaint, (int) width, Layout.Alignment.ALIGN_NORMAL, 1.0f, 0, false);

        // New API alternate
        //
        // StaticLayout.Builder builder = StaticLayout.Builder.obtain(mText, 0, mText.length(), mTextPaint, width)
        //        .setAlignment(Layout.Alignment.ALIGN_NORMAL)
        //        .setLineSpacing(1, 0) // multiplier, add
        //        .setIncludePad(false);
        // mStaticLayout = builder.build();
    }

    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        // Tell the parent layout how big this view would like to be
        // but still respect any requirements (measure specs) that are passed down.

        // determine the width
        int width;
        int widthMode = MeasureSpec.getMode(widthMeasureSpec);
        int widthRequirement = MeasureSpec.getSize(widthMeasureSpec);
        if (widthMode == MeasureSpec.EXACTLY) {
            width = widthRequirement;
        } else {
            width = mStaticLayout.getWidth() + getPaddingLeft() + getPaddingRight();
            if (widthMode == MeasureSpec.AT_MOST) {
                if (width > widthRequirement) {
                    width = widthRequirement;
                    // too long for a single line so relayout as multiline
                    mStaticLayout = new StaticLayout(mText, mTextPaint, width, Layout.Alignment.ALIGN_NORMAL, 1.0f, 0, false);
                }
            }
        }

        // determine the height
        int height;
        int heightMode = MeasureSpec.getMode(heightMeasureSpec);
        int heightRequirement = MeasureSpec.getSize(heightMeasureSpec);
        if (heightMode == MeasureSpec.EXACTLY) {
            height = heightRequirement;
        } else {
            height = mStaticLayout.getHeight() + getPaddingTop() + getPaddingBottom();
            if (heightMode == MeasureSpec.AT_MOST) {
                height = Math.min(height, heightRequirement);
            }
        }

        // Required call: set width and height
        setMeasuredDimension(width, height);
    }

    @Override
    protected void onDraw(Canvas canvas) {
        super.onDraw(canvas);
        // do as little as possible inside onDraw to improve performance

        // draw the text on the canvas after adjusting for padding
        canvas.save();
        canvas.translate(getPaddingLeft(), getPaddingTop());
        mStaticLayout.draw(canvas);
        canvas.restore();
    }
}
Suragch
źródło