FontFamily.h revision a5bb91190edbea0be0e78a8511b3c920b6e99c4e
19cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien/* 29cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien * Copyright (C) 2013 The Android Open Source Project 39cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien * 49cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien * Licensed under the Apache License, Version 2.0 (the "License"); 59cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien * you may not use this file except in compliance with the License. 69cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien * You may obtain a copy of the License at 79cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien * 89cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien * http://www.apache.org/licenses/LICENSE-2.0 99cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien * 109cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien * Unless required by applicable law or agreed to in writing, software 119cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien * distributed under the License is distributed on an "AS IS" BASIS, 129cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 139cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien * See the License for the specific language governing permissions and 149cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien * limitations under the License. 159cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien */ 169cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien 179cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien#ifndef MINIKIN_FONT_FAMILY_H 189cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien#define MINIKIN_FONT_FAMILY_H 199cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien 209cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien#include <vector> 21a5bb91190edbea0be0e78a8511b3c920b6e99c4eBehdad Esfahbod#include <string> 229cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien 234d4e6bc8118d15542f1f2a9218f0f7a91a29474fRaph Levien#include <utils/TypeHelpers.h> 244d4e6bc8118d15542f1f2a9218f0f7a91a29474fRaph Levien 25b80c1f19c58b927820a8a24bf2218e5645724608Raph Levien#include <minikin/MinikinRefCounted.h> 26b80c1f19c58b927820a8a24bf2218e5645724608Raph Levien 279cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Leviennamespace android { 289cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien 297b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levienclass MinikinFont; 307b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien 317b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien// FontLanguage is a compact representation of a bcp-47 language tag. It 327b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien// does not capture all possible information, only what directly affects 337b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien// font rendering. 347b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levienclass FontLanguage { 357b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien friend class FontStyle; 367b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levienpublic: 377b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien FontLanguage() : mBits(0) { } 387b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien 397b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien // Parse from string 407b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien FontLanguage(const char* buf, size_t size); 417b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien 427b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien bool operator==(const FontLanguage other) const { return mBits == other.mBits; } 43a5bb91190edbea0be0e78a8511b3c920b6e99c4eBehdad Esfahbod operator bool() const { return mBits != 0; } 44a5bb91190edbea0be0e78a8511b3c920b6e99c4eBehdad Esfahbod 45a5bb91190edbea0be0e78a8511b3c920b6e99c4eBehdad Esfahbod std::string getString() const; 467b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien 477b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien // 0 = no match, 1 = language matches, 2 = language and script match 487b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien int match(const FontLanguage other) const; 497b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien 507b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levienprivate: 517b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien explicit FontLanguage(uint32_t bits) : mBits(bits) { } 527b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien 537b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien uint32_t bits() const { return mBits; } 547b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien 557b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien static const uint32_t kBaseLangMask = 0xffff; 567b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien static const uint32_t kScriptMask = (1 << 18) - (1 << 16); 577b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien static const uint32_t kHansFlag = 1 << 16; 587b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien static const uint32_t kHantFlag = 1 << 17; 597b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien uint32_t mBits; 607b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien}; 617b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien 629cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien// FontStyle represents all style information needed to select an actual font 639cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien// from a collection. The implementation is packed into a single 32-bit word 649cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien// so it can be efficiently copied, embedded in other objects, etc. 659cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levienclass FontStyle { 669cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levienpublic: 679cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien FontStyle(int weight = 4, bool italic = false) { 689cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien bits = (weight & kWeightMask) | (italic ? kItalicMask : 0); 699cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien } 707b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien FontStyle(FontLanguage lang, int variant = 0, int weight = 4, bool italic = false) { 717b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien bits = (weight & kWeightMask) | (italic ? kItalicMask : 0) 727b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien | (variant << kVariantShift) | (lang.bits() << kLangShift); 737b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien } 744d4e6bc8118d15542f1f2a9218f0f7a91a29474fRaph Levien int getWeight() const { return bits & kWeightMask; } 754d4e6bc8118d15542f1f2a9218f0f7a91a29474fRaph Levien bool getItalic() const { return (bits & kItalicMask) != 0; } 767b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien int getVariant() const { return (bits >> kVariantShift) & kVariantMask; } 777b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien FontLanguage getLanguage() const { return FontLanguage(bits >> kLangShift); } 787b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien 794d4e6bc8118d15542f1f2a9218f0f7a91a29474fRaph Levien bool operator==(const FontStyle other) const { return bits == other.bits; } 804d4e6bc8118d15542f1f2a9218f0f7a91a29474fRaph Levien 814d4e6bc8118d15542f1f2a9218f0f7a91a29474fRaph Levien hash_t hash() const { return bits; } 829cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levienprivate: 837b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien static const uint32_t kWeightMask = (1 << 4) - 1; 847b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien static const uint32_t kItalicMask = 1 << 4; 857b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien static const int kVariantShift = 5; 867b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien static const uint32_t kVariantMask = (1 << 2) - 1; 877b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien static const int kLangShift = 7; 889cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien uint32_t bits; 899cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien}; 909cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien 917b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levienenum FontVariant { 927b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien VARIANT_DEFAULT = 0, 937b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien VARIANT_COMPACT = 1, 947b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien VARIANT_ELEGANT = 2, 957b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien}; 967b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien 974d4e6bc8118d15542f1f2a9218f0f7a91a29474fRaph Levieninline hash_t hash_type(const FontStyle &style) { 984d4e6bc8118d15542f1f2a9218f0f7a91a29474fRaph Levien return style.hash(); 994d4e6bc8118d15542f1f2a9218f0f7a91a29474fRaph Levien} 1004d4e6bc8118d15542f1f2a9218f0f7a91a29474fRaph Levien 1019a5f713add8cfb91ac2c9ed5c917309053201ab6Raph Levien// attributes representing transforms (fake bold, fake italic) to match styles 1029a5f713add8cfb91ac2c9ed5c917309053201ab6Raph Levienclass FontFakery { 1039a5f713add8cfb91ac2c9ed5c917309053201ab6Raph Levienpublic: 1049a5f713add8cfb91ac2c9ed5c917309053201ab6Raph Levien FontFakery() : mFakeBold(false), mFakeItalic(false) { } 1059a5f713add8cfb91ac2c9ed5c917309053201ab6Raph Levien FontFakery(bool fakeBold, bool fakeItalic) : mFakeBold(fakeBold), mFakeItalic(fakeItalic) { } 1069a5f713add8cfb91ac2c9ed5c917309053201ab6Raph Levien // TODO: want to support graded fake bolding 1079a5f713add8cfb91ac2c9ed5c917309053201ab6Raph Levien bool isFakeBold() { return mFakeBold; } 1089a5f713add8cfb91ac2c9ed5c917309053201ab6Raph Levien bool isFakeItalic() { return mFakeItalic; } 1099a5f713add8cfb91ac2c9ed5c917309053201ab6Raph Levienprivate: 1109a5f713add8cfb91ac2c9ed5c917309053201ab6Raph Levien bool mFakeBold; 1119a5f713add8cfb91ac2c9ed5c917309053201ab6Raph Levien bool mFakeItalic; 1129a5f713add8cfb91ac2c9ed5c917309053201ab6Raph Levien}; 1139a5f713add8cfb91ac2c9ed5c917309053201ab6Raph Levien 1149a5f713add8cfb91ac2c9ed5c917309053201ab6Raph Levienstruct FakedFont { 1159a5f713add8cfb91ac2c9ed5c917309053201ab6Raph Levien // ownership is the enclosing FontCollection 1169a5f713add8cfb91ac2c9ed5c917309053201ab6Raph Levien MinikinFont* font; 1179a5f713add8cfb91ac2c9ed5c917309053201ab6Raph Levien FontFakery fakery; 1189a5f713add8cfb91ac2c9ed5c917309053201ab6Raph Levien}; 1199a5f713add8cfb91ac2c9ed5c917309053201ab6Raph Levien 120b80c1f19c58b927820a8a24bf2218e5645724608Raph Levienclass FontFamily : public MinikinRefCounted { 1219cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levienpublic: 1227b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien FontFamily() { } 1237b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien 1247b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien FontFamily(FontLanguage lang, int variant) : mLang(lang), mVariant(variant) { 1257b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien } 1267b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien 127b80c1f19c58b927820a8a24bf2218e5645724608Raph Levien ~FontFamily(); 128b80c1f19c58b927820a8a24bf2218e5645724608Raph Levien 1299cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien // Add font to family, extracting style information from the font 130bcc3dc5a2591a95a57e379e27cbad69c18e91e67Raph Levien bool addFont(MinikinFont* typeface); 1319cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien 132bcc3dc5a2591a95a57e379e27cbad69c18e91e67Raph Levien void addFont(MinikinFont* typeface, FontStyle style); 1339a5f713add8cfb91ac2c9ed5c917309053201ab6Raph Levien FakedFont getClosestMatch(FontStyle style) const; 1349cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien 1357b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien FontLanguage lang() const { return mLang; } 1367b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien int variant() const { return mVariant; } 1377b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien 1389cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien // API's for enumerating the fonts in a family. These don't guarantee any particular order 1399cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien size_t getNumFonts() const; 140bcc3dc5a2591a95a57e379e27cbad69c18e91e67Raph Levien MinikinFont* getFont(size_t index) const; 1419cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien FontStyle getStyle(size_t index) const; 1429cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levienprivate: 143c31e3883456e018d742e9f29815ba5ff8b315ea1Raph Levien void addFontLocked(MinikinFont* typeface, FontStyle style); 144c31e3883456e018d742e9f29815ba5ff8b315ea1Raph Levien 1459cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien class Font { 1469cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien public: 147bcc3dc5a2591a95a57e379e27cbad69c18e91e67Raph Levien Font(MinikinFont* typeface, FontStyle style) : 1489cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien typeface(typeface), style(style) { } 149bcc3dc5a2591a95a57e379e27cbad69c18e91e67Raph Levien MinikinFont* typeface; 1509cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien FontStyle style; 1519cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien }; 1527b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien FontLanguage mLang; 1537b221d97b7b64dc5ce457e19666d55d042e22e62Raph Levien int mVariant; 1549cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien std::vector<Font> mFonts; 1559cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien}; 1569cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien 1579cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien} // namespace android 1589cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien 1599cc9bbe1461f359f0b27c5e7645c17dda001ab1dRaph Levien#endif // MINIKIN_FONT_FAMILY_H 160