FontRenderer.h revision 65ef909776c03417d8b597738da54ca211e37e4f
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,
50694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy            int numGlyphs, int x, int y);
5151769a68a5cb34e9564740c6a854fcb93018789dRomain Guy    /**
5251769a68a5cb34e9564740c6a854fcb93018789dRomain Guy     * Creates a new font associated with the specified font state.
5351769a68a5cb34e9564740c6a854fcb93018789dRomain Guy     */
54694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    static Font* create(FontRenderer* state, uint32_t fontId, float fontSize);
55694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
56694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guyprotected:
57694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    friend class FontRenderer;
58694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
59694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    struct CachedGlyphInfo {
60694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        // Has the cache been invalidated?
61694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        bool mIsValid;
62694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        // Location of the cached glyph in the bitmap
63694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        // in case we need to resize the texture
64694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        uint32_t mBitmapWidth;
65694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        uint32_t mBitmapHeight;
66694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        // Also cache texture coords for the quad
67694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        float mBitmapMinU;
68694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        float mBitmapMinV;
69694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        float mBitmapMaxU;
70694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        float mBitmapMaxV;
71694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        // Minimize how much we call freetype
72694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        uint32_t mGlyphIndex;
73694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        uint32_t mAdvanceX;
74694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        uint32_t mAdvanceY;
75694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        // Values below contain a glyph's origin in the bitmap
76694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        uint32_t mBitmapLeft;
77694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        uint32_t mBitmapTop;
78694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    };
79694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
80694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    Font(FontRenderer* state, uint32_t fontId, float fontSize);
81694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
82694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    DefaultKeyedVector<int32_t, CachedGlyphInfo*> mCachedGlyphs;
83694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
84bd0e6aa0ff0bd8b376772c3e23513a6021bdda87Romain Guy    void invalidateTextureCache();
85bd0e6aa0ff0bd8b376772c3e23513a6021bdda87Romain Guy
8665ef909776c03417d8b597738da54ca211e37e4fAlex Sakhartchouk    CachedGlyphInfo* cacheGlyph(SkPaint* paint, int32_t glyph);
87694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    void updateGlyphCache(SkPaint* paint, const SkGlyph& skiaGlyph, CachedGlyphInfo *glyph);
88694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    void drawCachedGlyph(CachedGlyphInfo *glyph, int x, int y);
89bd0e6aa0ff0bd8b376772c3e23513a6021bdda87Romain Guy
9065ef909776c03417d8b597738da54ca211e37e4fAlex Sakhartchouk    CachedGlyphInfo* getCachedUTFChar(SkPaint* paint, int32_t utfChar);
9165ef909776c03417d8b597738da54ca211e37e4fAlex Sakhartchouk
92bd0e6aa0ff0bd8b376772c3e23513a6021bdda87Romain Guy    FontRenderer* mState;
93bd0e6aa0ff0bd8b376772c3e23513a6021bdda87Romain Guy    uint32_t mFontId;
94bd0e6aa0ff0bd8b376772c3e23513a6021bdda87Romain Guy    float mFontSize;
95694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy};
96694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
97694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guyclass FontRenderer {
98694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guypublic:
99694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    FontRenderer();
100694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    ~FontRenderer();
101694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
102694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    void init();
103694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    void deinit();
104694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
10565ef909776c03417d8b597738da54ca211e37e4fAlex Sakhartchouk    void setFont(SkPaint* paint, uint32_t fontId, float fontSize);
10651769a68a5cb34e9564740c6a854fcb93018789dRomain Guy    void renderText(SkPaint* paint, const Rect* clip, const char *text, uint32_t startIndex,
10751769a68a5cb34e9564740c6a854fcb93018789dRomain Guy            uint32_t len, int numGlyphs, int x, int y);
108694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
109694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    GLuint getTexture() {
110694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        checkInit();
111694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        return mTextureId;
112694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    }
113694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
114694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guyprotected:
115694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    friend class Font;
116694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
117694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    struct CacheTextureLine {
118694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        uint16_t mMaxHeight;
119694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        uint16_t mMaxWidth;
120694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        uint32_t mCurrentRow;
121694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        uint32_t mCurrentCol;
1229b9902ddbb01548f4a0199087b7035e7c10b2ae7Alex Sakhartchouk        bool     mDirty;
123694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
12451769a68a5cb34e9564740c6a854fcb93018789dRomain Guy        CacheTextureLine(uint16_t maxWidth, uint16_t maxHeight, uint32_t currentRow,
125694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy                uint32_t currentCol):
12651769a68a5cb34e9564740c6a854fcb93018789dRomain Guy                    mMaxHeight(maxHeight),
12751769a68a5cb34e9564740c6a854fcb93018789dRomain Guy                    mMaxWidth(maxWidth),
12851769a68a5cb34e9564740c6a854fcb93018789dRomain Guy                    mCurrentRow(currentRow),
1299b9902ddbb01548f4a0199087b7035e7c10b2ae7Alex Sakhartchouk                    mCurrentCol(currentCol),
1309b9902ddbb01548f4a0199087b7035e7c10b2ae7Alex Sakhartchouk                    mDirty(false) {
131694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        }
132694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
133694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        bool fitBitmap(const SkGlyph& glyph, uint32_t *retOriginX, uint32_t *retOriginY) {
134694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy            if (glyph.fHeight > mMaxHeight) {
135694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy                return false;
136694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy            }
137694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
138694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy            if (mCurrentCol + glyph.fWidth < mMaxWidth) {
139694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy                *retOriginX = mCurrentCol;
140694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy                *retOriginY = mCurrentRow;
141694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy                mCurrentCol += glyph.fWidth;
1429b9902ddbb01548f4a0199087b7035e7c10b2ae7Alex Sakhartchouk                mDirty = true;
143694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy                return true;
144694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy            }
145694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
146694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy            return false;
147694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        }
148694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    };
149694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
150694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    uint32_t getCacheWidth() const {
151694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        return mCacheWidth;
152694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    }
153694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
154694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    uint32_t getCacheHeight() const {
155694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy        return mCacheHeight;
156694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    }
157694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
158694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    void initTextTexture();
159694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    bool cacheBitmap(const SkGlyph& glyph, uint32_t *retOriginX, uint32_t *retOriginY);
160694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
161694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    void flushAllAndInvalidate();
162694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    void initVertexArrayBuffers();
163694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
164694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    void checkInit();
165694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
16665ef909776c03417d8b597738da54ca211e37e4fAlex Sakhartchouk    String16 mLatinPrecache;
16765ef909776c03417d8b597738da54ca211e37e4fAlex Sakhartchouk    void precacheLatin(SkPaint* paint);
16865ef909776c03417d8b597738da54ca211e37e4fAlex Sakhartchouk
169694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    void issueDrawCommand();
170694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    void appendMeshQuad(float x1, float y1, float z1, float u1, float v1, float x2, float y2,
171694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy            float z2, float u2, float v2, float x3, float y3, float z3, float u3, float v3,
172694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy            float x4, float y4, float z4, float u4, float v4);
173694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
174694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    uint32_t mCacheWidth;
175694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    uint32_t mCacheHeight;
176694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
177694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    Vector<CacheTextureLine*> mCacheLines;
17865ef909776c03417d8b597738da54ca211e37e4fAlex Sakhartchouk    uint32_t getRemainingCacheCapacity();
179694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
18009147fbdc8206a0cac78bfe9083e7e15b3c5689cRomain Guy    Font* mCurrentFont;
181694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    Vector<Font*> mActiveFonts;
182694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
183694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    // Texture to cache glyph bitmaps
184694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    unsigned char* mTextTexture;
185694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    GLuint mTextureId;
1869b9902ddbb01548f4a0199087b7035e7c10b2ae7Alex Sakhartchouk    void checkTextureUpdate();
187694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    bool mUploadTexture;
188694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
189694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    // Pointer to vertex data to speed up frame to frame work
190694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    float *mTextMeshPtr;
191694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    uint32_t mCurrentQuadIndex;
192694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    uint32_t mMaxNumberOfQuads;
193694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
194694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    uint32_t mIndexBufferID;
195694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
19609147fbdc8206a0cac78bfe9083e7e15b3c5689cRomain Guy    const Rect* mClip;
19709147fbdc8206a0cac78bfe9083e7e15b3c5689cRomain Guy
198694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy    bool mInitialized;
199694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy};
200694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
201694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy}; // namespace uirenderer
202694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy}; // namespace android
203694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy
204694b519ac647fe998fd396fe0784cc8e179aadc4Romain Guy#endif // ANDROID_UI_FONT_RENDERER_H
205