FontRenderer.h revision 89a524ac2d4a36739e51b01b336c0bade77e2ee0
1694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy/*
2694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy * Copyright (C) 2010 The Android Open Source Project
3694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy *
4694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy * Licensed under the Apache License, Version 2.0 (the "License");
5694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy * you may not use this file except in compliance with the License.
6694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy * You may obtain a copy of the License at
7694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy *
8694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy *      http://www.apache.org/licenses/LICENSE-2.0
9694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy *
10694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy * Unless required by applicable law or agreed to in writing, software
11694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy * distributed under the License is distributed on an "AS IS" BASIS,
12694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy * See the License for the specific language governing permissions and
14694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy * limitations under the License.
15694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy */
16694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
17694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy#ifndef ANDROID_UI_FONT_RENDERER_H
18694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy#define ANDROID_UI_FONT_RENDERER_H
19694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
20694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy#include <utils/String8.h>
2165ef909776c03417d8b597738da54ca211e37e4fAlex Sakhartchouk#include <utils/String16.h>
22694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy#include <utils/Vector.h>
23694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy#include <utils/KeyedVector.h>
24694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
25694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy#include <SkScalerContext.h>
26694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy#include <SkPaint.h>
27694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
28694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy#include <GLES2/gl2.h>
29694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
3009147fbdc8206a0cac78bfe9083e7e15b3c5689cRomain Guy#include "Rect.h"
3151769a68a5cb34e9564740c6a854fcb93018789dRomain Guy#include "Properties.h"
3209147fbdc8206a0cac78bfe9083e7e15b3c5689cRomain Guy
33694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guynamespace android {
34694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guynamespace uirenderer {
35694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
36694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guyclass FontRenderer;
37694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
3851769a68a5cb34e9564740c6a854fcb93018789dRomain Guy/**
3951769a68a5cb34e9564740c6a854fcb93018789dRomain Guy * Represents a font, defined by a Skia font id and a font size. A font is used
4051769a68a5cb34e9564740c6a854fcb93018789dRomain Guy * to generate glyphs and cache them in the FontState.
4151769a68a5cb34e9564740c6a854fcb93018789dRomain Guy */
42694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guyclass Font {
43694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guypublic:
44694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    ~Font();
45694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
4651769a68a5cb34e9564740c6a854fcb93018789dRomain Guy    /**
4751769a68a5cb34e9564740c6a854fcb93018789dRomain Guy     * Renders the specified string of text.
4851769a68a5cb34e9564740c6a854fcb93018789dRomain Guy     */
4951769a68a5cb34e9564740c6a854fcb93018789dRomain Guy    void renderUTF(SkPaint* paint, const char *text, uint32_t start, uint32_t len,
5089a524ac2d4a36739e51b01b336c0bade77e2ee0Alex Sakhartchouk                     int numGlyphs, int x, int y,
5189a524ac2d4a36739e51b01b336c0bade77e2ee0Alex Sakhartchouk                     uint8_t *bitmap = NULL, uint32_t bitmapW = 0, uint32_t bitmapH = 0);
5251769a68a5cb34e9564740c6a854fcb93018789dRomain Guy    /**
5351769a68a5cb34e9564740c6a854fcb93018789dRomain Guy     * Creates a new font associated with the specified font state.
5451769a68a5cb34e9564740c6a854fcb93018789dRomain Guy     */
55694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    static Font* create(FontRenderer* state, uint32_t fontId, float fontSize);
56694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
57694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guyprotected:
58694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    friend class FontRenderer;
59694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
60694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    struct CachedGlyphInfo {
61694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        // Has the cache been invalidated?
62694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        bool mIsValid;
63694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        // Location of the cached glyph in the bitmap
6489a524ac2d4a36739e51b01b336c0bade77e2ee0Alex Sakhartchouk        // in case we need to resize the texture or
6589a524ac2d4a36739e51b01b336c0bade77e2ee0Alex Sakhartchouk        // render to bitmap
6689a524ac2d4a36739e51b01b336c0bade77e2ee0Alex Sakhartchouk        uint32_t mStartX;
6789a524ac2d4a36739e51b01b336c0bade77e2ee0Alex Sakhartchouk        uint32_t mStartY;
68694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        uint32_t mBitmapWidth;
69694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        uint32_t mBitmapHeight;
70694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        // Also cache texture coords for the quad
71694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        float mBitmapMinU;
72694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        float mBitmapMinV;
73694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        float mBitmapMaxU;
74694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        float mBitmapMaxV;
75694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        // Minimize how much we call freetype
76694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        uint32_t mGlyphIndex;
77694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        uint32_t mAdvanceX;
78694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        uint32_t mAdvanceY;
79694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        // Values below contain a glyph's origin in the bitmap
8089a524ac2d4a36739e51b01b336c0bade77e2ee0Alex Sakhartchouk        int32_t mBitmapLeft;
8189a524ac2d4a36739e51b01b336c0bade77e2ee0Alex Sakhartchouk        int32_t mBitmapTop;
82694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    };
83694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
84694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    Font(FontRenderer* state, uint32_t fontId, float fontSize);
85694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
86694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    DefaultKeyedVector<int32_t, CachedGlyphInfo*> mCachedGlyphs;
87694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
88bd0e6aa0ff0bd8b376772c3e23513a6021bdda87Romain Guy    void invalidateTextureCache();
89bd0e6aa0ff0bd8b376772c3e23513a6021bdda87Romain Guy
9065ef909776c03417d8b597738da54ca211e37e4fAlex Sakhartchouk    CachedGlyphInfo* cacheGlyph(SkPaint* paint, int32_t glyph);
91694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    void updateGlyphCache(SkPaint* paint, const SkGlyph& skiaGlyph, CachedGlyphInfo *glyph);
92694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    void drawCachedGlyph(CachedGlyphInfo *glyph, int x, int y);
9389a524ac2d4a36739e51b01b336c0bade77e2ee0Alex Sakhartchouk    void drawCachedGlyph(CachedGlyphInfo *glyph, int x, int y,
9489a524ac2d4a36739e51b01b336c0bade77e2ee0Alex Sakhartchouk                          uint8_t *bitmap, uint32_t bitmapW, uint32_t bitmapH);
95bd0e6aa0ff0bd8b376772c3e23513a6021bdda87Romain Guy
9665ef909776c03417d8b597738da54ca211e37e4fAlex Sakhartchouk    CachedGlyphInfo* getCachedUTFChar(SkPaint* paint, int32_t utfChar);
9765ef909776c03417d8b597738da54ca211e37e4fAlex Sakhartchouk
98bd0e6aa0ff0bd8b376772c3e23513a6021bdda87Romain Guy    FontRenderer* mState;
99bd0e6aa0ff0bd8b376772c3e23513a6021bdda87Romain Guy    uint32_t mFontId;
100bd0e6aa0ff0bd8b376772c3e23513a6021bdda87Romain Guy    float mFontSize;
101694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy};
102694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
103694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guyclass FontRenderer {
104694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guypublic:
105694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    FontRenderer();
106694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    ~FontRenderer();
107694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
108694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    void init();
109694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    void deinit();
110694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
11165ef909776c03417d8b597738da54ca211e37e4fAlex Sakhartchouk    void setFont(SkPaint* paint, uint32_t fontId, float fontSize);
11251769a68a5cb34e9564740c6a854fcb93018789dRomain Guy    void renderText(SkPaint* paint, const Rect* clip, const char *text, uint32_t startIndex,
11351769a68a5cb34e9564740c6a854fcb93018789dRomain Guy            uint32_t len, int numGlyphs, int x, int y);
114694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
115694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    GLuint getTexture() {
116694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        checkInit();
117694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        return mTextureId;
118694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    }
119694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
120694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guyprotected:
121694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    friend class Font;
122694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
123694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    struct CacheTextureLine {
124694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        uint16_t mMaxHeight;
125694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        uint16_t mMaxWidth;
126694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        uint32_t mCurrentRow;
127694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        uint32_t mCurrentCol;
1289b9902ddbb01548f4a0199087b7035e7c10b2ae7Alex Sakhartchouk        bool     mDirty;
129694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
13051769a68a5cb34e9564740c6a854fcb93018789dRomain Guy        CacheTextureLine(uint16_t maxWidth, uint16_t maxHeight, uint32_t currentRow,
131694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy                uint32_t currentCol):
13251769a68a5cb34e9564740c6a854fcb93018789dRomain Guy                    mMaxHeight(maxHeight),
13351769a68a5cb34e9564740c6a854fcb93018789dRomain Guy                    mMaxWidth(maxWidth),
13451769a68a5cb34e9564740c6a854fcb93018789dRomain Guy                    mCurrentRow(currentRow),
1359b9902ddbb01548f4a0199087b7035e7c10b2ae7Alex Sakhartchouk                    mCurrentCol(currentCol),
1369b9902ddbb01548f4a0199087b7035e7c10b2ae7Alex Sakhartchouk                    mDirty(false) {
137694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        }
138694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
139694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        bool fitBitmap(const SkGlyph& glyph, uint32_t *retOriginX, uint32_t *retOriginY) {
140694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy            if (glyph.fHeight > mMaxHeight) {
141694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy                return false;
142694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy            }
143694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
144694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy            if (mCurrentCol + glyph.fWidth < mMaxWidth) {
145694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy                *retOriginX = mCurrentCol;
146694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy                *retOriginY = mCurrentRow;
147694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy                mCurrentCol += glyph.fWidth;
1489b9902ddbb01548f4a0199087b7035e7c10b2ae7Alex Sakhartchouk                mDirty = true;
149694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy                return true;
150694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy            }
151694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
152694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy            return false;
153694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        }
154694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    };
155694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
156694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    uint32_t getCacheWidth() const {
157694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        return mCacheWidth;
158694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    }
159694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
160694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    uint32_t getCacheHeight() const {
161694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        return mCacheHeight;
162694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    }
163694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
164694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    void initTextTexture();
165694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    bool cacheBitmap(const SkGlyph& glyph, uint32_t *retOriginX, uint32_t *retOriginY);
166694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
167694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    void flushAllAndInvalidate();
168694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    void initVertexArrayBuffers();
169694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
170694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    void checkInit();
171694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
17265ef909776c03417d8b597738da54ca211e37e4fAlex Sakhartchouk    String16 mLatinPrecache;
17365ef909776c03417d8b597738da54ca211e37e4fAlex Sakhartchouk    void precacheLatin(SkPaint* paint);
17465ef909776c03417d8b597738da54ca211e37e4fAlex Sakhartchouk
175694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    void issueDrawCommand();
176694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    void appendMeshQuad(float x1, float y1, float z1, float u1, float v1, float x2, float y2,
177694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy            float z2, float u2, float v2, float x3, float y3, float z3, float u3, float v3,
178694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy            float x4, float y4, float z4, float u4, float v4);
179694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
180694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    uint32_t mCacheWidth;
181694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    uint32_t mCacheHeight;
182694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
183694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    Vector<CacheTextureLine*> mCacheLines;
18465ef909776c03417d8b597738da54ca211e37e4fAlex Sakhartchouk    uint32_t getRemainingCacheCapacity();
185694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
18609147fbdc8206a0cac78bfe9083e7e15b3c5689cRomain Guy    Font* mCurrentFont;
187694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    Vector<Font*> mActiveFonts;
188694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
189694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    // Texture to cache glyph bitmaps
19089a524ac2d4a36739e51b01b336c0bade77e2ee0Alex Sakhartchouk    uint8_t* mTextTexture;
19189a524ac2d4a36739e51b01b336c0bade77e2ee0Alex Sakhartchouk    const uint8_t* getTextTextureData() const {
19289a524ac2d4a36739e51b01b336c0bade77e2ee0Alex Sakhartchouk        return mTextTexture;
19389a524ac2d4a36739e51b01b336c0bade77e2ee0Alex Sakhartchouk    }
194694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    GLuint mTextureId;
1959b9902ddbb01548f4a0199087b7035e7c10b2ae7Alex Sakhartchouk    void checkTextureUpdate();
196694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    bool mUploadTexture;
197694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
198694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    // Pointer to vertex data to speed up frame to frame work
199694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    float *mTextMeshPtr;
200694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    uint32_t mCurrentQuadIndex;
201694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    uint32_t mMaxNumberOfQuads;
202694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
203694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    uint32_t mIndexBufferID;
204694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
20509147fbdc8206a0cac78bfe9083e7e15b3c5689cRomain Guy    const Rect* mClip;
20609147fbdc8206a0cac78bfe9083e7e15b3c5689cRomain Guy
207694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    bool mInitialized;
20889a524ac2d4a36739e51b01b336c0bade77e2ee0Alex Sakhartchouk
20989a524ac2d4a36739e51b01b336c0bade77e2ee0Alex Sakhartchouk    void computeGaussianWeights(float* weights, int32_t radius);
21089a524ac2d4a36739e51b01b336c0bade77e2ee0Alex Sakhartchouk    void horizontalBlur(float* weights, int32_t radius, const uint8_t *source, uint8_t *dest,
21189a524ac2d4a36739e51b01b336c0bade77e2ee0Alex Sakhartchouk                         int32_t width, int32_t height);
21289a524ac2d4a36739e51b01b336c0bade77e2ee0Alex Sakhartchouk    void verticalBlur(float* weights, int32_t radius, const uint8_t *source, uint8_t *dest,
21389a524ac2d4a36739e51b01b336c0bade77e2ee0Alex Sakhartchouk                         int32_t width, int32_t height);
21489a524ac2d4a36739e51b01b336c0bade77e2ee0Alex Sakhartchouk    void blurImage(uint8_t* image, int32_t width, int32_t height, int32_t radius);
215694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy};
216694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
217694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy}; // namespace uirenderer
218694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy}; // namespace android
219694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
220694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy#endif // ANDROID_UI_FONT_RENDERER_H
221