繪制基礎(chǔ)(四)- 繪制字體

平時(shí)項(xiàng)目里會(huì)涉及到一些文字繪制,這里記錄下自己學(xué)的的。

先來一張網(wǎng)上找的圖,簡單介紹下android是按啥規(guī)則繪制字體的:


FontMetrics

這個(gè)需要結(jié)合Paint.FontMetrics這個(gè)類來看:

    /**
     * Class that describes the various metrics for a font at a given text size.
     * Remember, Y values increase going down, so those values will be positive,
     * and values that measure distances going up will be negative. This class
     * is returned by getFontMetrics().
     */
    public static class FontMetrics {
        /**
         * The maximum distance above the baseline for the tallest glyph in
         * the font at a given text size.
         */
        public float   top;
        /**
         * The recommended distance above the baseline for singled spaced text.
         */
        public float   ascent;
        /**
         * The recommended distance below the baseline for singled spaced text.
         */
        public float   descent;
        /**
         * The maximum distance below the baseline for the lowest glyph in
         * the font at a given text size.
         */
        public float   bottom;
        /**
         * The recommended additional space to add between lines of text.
         */
        public float   leading;
    }

需要特別關(guān)注的就是BaseLine,像canvas.drawText()里頭的y,就是baseLine的位置,所以,有時(shí)候雖然我們繪制時(shí)參數(shù)y在居中位置,但是實(shí)際體現(xiàn)的效果確不居中,這就是因?yàn)槲淖謺r(shí)基于baseLine來繪制的。這個(gè)時(shí)候的y,就需要根據(jù)文字來特地計(jì)算:

public class SportsView extends View {

    private final static float RING_WIDTH = UiUtils.dpToPixel(20);
    private final static float RADIUS = UiUtils.dpToPixel(150);
    private Paint paint = new Paint(Paint.ANTI_ALIAS_FLAG);
    private Rect rect = new Rect();
    private Paint.FontMetrics fontMetrics = new Paint.FontMetrics();

    public SportsView(Context context, @Nullable AttributeSet attrs) {
        super(context, attrs);
    }

    {
        paint.setTextSize(UiUtils.dpToPixel(100));
        paint.setTextAlign(Paint.Align.CENTER);
    }

    @Override
    protected void onDraw(Canvas canvas) {
        super.onDraw(canvas);

        // 繪制環(huán)
        paint.setStyle(Paint.Style.STROKE);
        paint.setColor(Color.GREEN);
        paint.setStrokeWidth(RING_WIDTH);
        canvas.drawCircle(getWidth() / 2, getHeight() / 2, RADIUS, paint);

        // 繪制進(jìn)度條
        paint.setColor(Color.RED);
        paint.setStrokeCap(Paint.Cap.ROUND);
        canvas.drawArc(getWidth() / 2 - RADIUS, getHeight() / 2 - RADIUS,
                getWidth() / 2 + RADIUS, getHeight() / 2 + RADIUS,
                -90, 225, false, paint);

        paint.setStyle(Paint.Style.FILL);
//        // 讀取文字的實(shí)時(shí)繪制范圍,以用于后面如何對baseline進(jìn)行偏移
//        paint.getTextBounds("123", 0, "123".length(), rect);
//        int offset = (rect.top + rect.bottom) / 2;
//        // 會(huì)在baseLine上畫,baseline是一個(gè)穩(wěn)定線,因?yàn)槲淖值膟軸難以確定
//        // 文字橫向居中很容易,使用Center即可,但是縱向居中就比較麻煩了
//        // 雖然該方法會(huì)讓文字非常居中,但是,當(dāng)文字變化是,可能會(huì)跳動(dòng),因?yàn)槔L制范圍是根據(jù)文字內(nèi)容來計(jì)算的
//        canvas.drawText("123", getWidth() / 2, getHeight() / 2 - offset, paint);

          // 左對齊情況,如果希望文字完全貼邊,即去除文字訪問的左邊部分
//        canvas.drawText("123", - rect.left, 200, paint);

        paint.getFontMetrics(fontMetrics);
        // 這樣標(biāo)準(zhǔn)比較統(tǒng)一,不會(huì)讓文字跳動(dòng),適用于變動(dòng)場景
        float offset = (fontMetrics.ascent + fontMetrics.descent) / 2;
        canvas.drawText("123", getWidth() / 2, getHeight() / 2 - offset, paint);
    }
}
效果圖
總結(jié):文字繪制主要是BaseLine的應(yīng)用。
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • 用兩張圖告訴你,為什么你的 App 會(huì)卡頓? - Android - 掘金 Cover 有什么料? 從這篇文章中你...
    hw1212閱讀 14,048評論 2 59
  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 179,192評論 25 708
  • 一、概述 1. 四線格與基線 小時(shí)候,我們在剛開始學(xué)習(xí)寫字母時(shí),用的本子是四線格的,我們必須把字母按照規(guī)則寫在四線...
    addapp閱讀 8,041評論 2 17
  • 京東前端實(shí)習(xí)生面試經(jīng)驗(yàn):作為一個(gè)前端開發(fā)人員,CSS3和HTML5 必須熟練使用CSS3新特性:1、拋棄圖片的視覺...
    碎碎先生閱讀 231評論 0 0
  • 好看的臉太多,有趣的靈魂卻很少,我一直單身的理由卻是 :不將就 不想找一個(gè)一起吃飯,逛街,在床上的伴而已 無聊的人...
    伊古董閱讀 362評論 0 2

友情鏈接更多精彩內(nèi)容