1 /* 2 * Copyright (C) 2017 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 package android.content.res; 17 18 import android.annotation.NonNull; 19 import android.annotation.Nullable; 20 import android.graphics.Typeface; 21 import android.util.AttributeSet; 22 import android.util.Log; 23 import android.util.Xml; 24 25 import com.android.internal.R; 26 27 import org.xmlpull.v1.XmlPullParser; 28 import org.xmlpull.v1.XmlPullParserException; 29 30 import java.io.IOException; 31 import java.util.ArrayList; 32 import java.util.Arrays; 33 import java.util.List; 34 35 /** 36 * Parser for xml type font resources. 37 * @hide 38 */ 39 public class FontResourcesParser { 40 private static final String TAG = "FontResourcesParser"; 41 42 // A class represents single entry of font-family in xml file. 43 public interface FamilyResourceEntry {} 44 45 // A class represents font provider based font-family element in xml file. 46 public static final class ProviderResourceEntry implements FamilyResourceEntry { 47 private final @NonNull String mProviderAuthority; 48 private final @NonNull String mProviderPackage; 49 private final @NonNull String mQuery; 50 private final @Nullable String mSystemFontFamilyName; 51 private final @Nullable List<List<String>> mCerts; 52 ProviderResourceEntry(@onNull String authority, @NonNull String pkg, @NonNull String query, @Nullable List<List<String>> certs, @Nullable String systemFontFamilyName)53 public ProviderResourceEntry(@NonNull String authority, @NonNull String pkg, 54 @NonNull String query, @Nullable List<List<String>> certs, 55 @Nullable String systemFontFamilyName) { 56 mProviderAuthority = authority; 57 mProviderPackage = pkg; 58 mQuery = query; 59 mCerts = certs; 60 mSystemFontFamilyName = systemFontFamilyName; 61 } 62 getAuthority()63 public @NonNull String getAuthority() { 64 return mProviderAuthority; 65 } 66 getPackage()67 public @NonNull String getPackage() { 68 return mProviderPackage; 69 } 70 getQuery()71 public @NonNull String getQuery() { 72 return mQuery; 73 } 74 getSystemFontFamilyName()75 public @NonNull String getSystemFontFamilyName() { 76 return mSystemFontFamilyName; 77 } 78 getCerts()79 public @Nullable List<List<String>> getCerts() { 80 return mCerts; 81 } 82 } 83 84 // A class represents font element in xml file which points a file in resource. 85 public static final class FontFileResourceEntry { 86 public static final int RESOLVE_BY_FONT_TABLE = Typeface.RESOLVE_BY_FONT_TABLE; 87 public static final int UPRIGHT = 0; 88 public static final int ITALIC = 1; 89 90 private final @NonNull String mFileName; 91 private int mWeight; 92 private int mItalic; 93 private int mTtcIndex; 94 private String mVariationSettings; 95 private int mResourceId; 96 FontFileResourceEntry(@onNull String fileName, int weight, int italic, @Nullable String variationSettings, int ttcIndex)97 public FontFileResourceEntry(@NonNull String fileName, int weight, int italic, 98 @Nullable String variationSettings, int ttcIndex) { 99 mFileName = fileName; 100 mWeight = weight; 101 mItalic = italic; 102 mVariationSettings = variationSettings; 103 mTtcIndex = ttcIndex; 104 } 105 getFileName()106 public @NonNull String getFileName() { 107 return mFileName; 108 } 109 getWeight()110 public int getWeight() { 111 return mWeight; 112 } 113 getItalic()114 public int getItalic() { 115 return mItalic; 116 } 117 getVariationSettings()118 public @Nullable String getVariationSettings() { 119 return mVariationSettings; 120 } 121 getTtcIndex()122 public int getTtcIndex() { 123 return mTtcIndex; 124 } 125 } 126 127 // A class represents file based font-family element in xml file. 128 public static final class FontFamilyFilesResourceEntry implements FamilyResourceEntry { 129 private final @NonNull FontFileResourceEntry[] mEntries; 130 FontFamilyFilesResourceEntry(@onNull FontFileResourceEntry[] entries)131 public FontFamilyFilesResourceEntry(@NonNull FontFileResourceEntry[] entries) { 132 mEntries = entries; 133 } 134 getEntries()135 public @NonNull FontFileResourceEntry[] getEntries() { 136 return mEntries; 137 } 138 } 139 parse(XmlPullParser parser, Resources resources)140 public static @Nullable FamilyResourceEntry parse(XmlPullParser parser, Resources resources) 141 throws XmlPullParserException, IOException { 142 int type; 143 while ((type=parser.next()) != XmlPullParser.START_TAG 144 && type != XmlPullParser.END_DOCUMENT) { 145 // Empty loop. 146 } 147 148 if (type != XmlPullParser.START_TAG) { 149 throw new XmlPullParserException("No start tag found"); 150 } 151 return readFamilies(parser, resources); 152 } 153 readFamilies(XmlPullParser parser, Resources resources)154 private static @Nullable FamilyResourceEntry readFamilies(XmlPullParser parser, 155 Resources resources) throws XmlPullParserException, IOException { 156 parser.require(XmlPullParser.START_TAG, null, "font-family"); 157 String tag = parser.getName(); 158 FamilyResourceEntry result = null; 159 if (tag.equals("font-family")) { 160 return readFamily(parser, resources); 161 } else { 162 skip(parser); 163 Log.e(TAG, "Failed to find font-family tag"); 164 return null; 165 } 166 } 167 readFamily(XmlPullParser parser, Resources resources)168 private static @Nullable FamilyResourceEntry readFamily(XmlPullParser parser, 169 Resources resources) throws XmlPullParserException, IOException { 170 AttributeSet attrs = Xml.asAttributeSet(parser); 171 TypedArray array = resources.obtainAttributes(attrs, R.styleable.FontFamily); 172 String authority = array.getString(R.styleable.FontFamily_fontProviderAuthority); 173 String providerPackage = array.getString(R.styleable.FontFamily_fontProviderPackage); 174 String query = array.getString(R.styleable.FontFamily_fontProviderQuery); 175 int certsId = array.getResourceId(R.styleable.FontFamily_fontProviderCerts, 0); 176 String systemFontFamilyName = array.getString( 177 R.styleable.FontFamily_fontProviderSystemFontFamily); 178 array.recycle(); 179 if (authority != null && providerPackage != null && query != null) { 180 while (parser.next() != XmlPullParser.END_TAG) { 181 skip(parser); 182 } 183 List<List<String>> certs = null; 184 if (certsId != 0) { 185 TypedArray typedArray = resources.obtainTypedArray(certsId); 186 if (typedArray.length() > 0) { 187 certs = new ArrayList<>(); 188 boolean isArrayOfArrays = typedArray.getResourceId(0, 0) != 0; 189 if (isArrayOfArrays) { 190 for (int i = 0; i < typedArray.length(); i++) { 191 int certId = typedArray.getResourceId(i, 0); 192 String[] certsArray = resources.getStringArray(certId); 193 List<String> certsList = Arrays.asList(certsArray); 194 certs.add(certsList); 195 } 196 } else { 197 String[] certsArray = resources.getStringArray(certsId); 198 List<String> certsList = Arrays.asList(certsArray); 199 certs.add(certsList); 200 } 201 } 202 typedArray.recycle(); 203 } 204 return new ProviderResourceEntry( 205 authority, 206 providerPackage, 207 query, 208 certs, 209 systemFontFamilyName 210 ); 211 } 212 List<FontFileResourceEntry> fonts = new ArrayList<>(); 213 while (parser.next() != XmlPullParser.END_TAG) { 214 if (parser.getEventType() != XmlPullParser.START_TAG) continue; 215 String tag = parser.getName(); 216 if (tag.equals("font")) { 217 final FontFileResourceEntry entry = readFont(parser, resources); 218 if (entry != null) { 219 fonts.add(entry); 220 } 221 } else { 222 skip(parser); 223 } 224 } 225 if (fonts.isEmpty()) { 226 return null; 227 } 228 return new FontFamilyFilesResourceEntry(fonts.toArray( 229 new FontFileResourceEntry[fonts.size()])); 230 } 231 readFont(XmlPullParser parser, Resources resources)232 private static FontFileResourceEntry readFont(XmlPullParser parser, Resources resources) 233 throws XmlPullParserException, IOException { 234 AttributeSet attrs = Xml.asAttributeSet(parser); 235 TypedArray array = resources.obtainAttributes(attrs, R.styleable.FontFamilyFont); 236 int weight = array.getInt(R.styleable.FontFamilyFont_fontWeight, 237 Typeface.RESOLVE_BY_FONT_TABLE); 238 int italic = array.getInt(R.styleable.FontFamilyFont_fontStyle, 239 FontFileResourceEntry.RESOLVE_BY_FONT_TABLE); 240 String variationSettings = array.getString( 241 R.styleable.FontFamilyFont_fontVariationSettings); 242 int ttcIndex = array.getInt(R.styleable.FontFamilyFont_ttcIndex, 0); 243 String filename = array.getString(R.styleable.FontFamilyFont_font); 244 array.recycle(); 245 while (parser.next() != XmlPullParser.END_TAG) { 246 skip(parser); 247 } 248 if (filename == null) { 249 return null; 250 } 251 return new FontFileResourceEntry(filename, weight, italic, variationSettings, ttcIndex); 252 } 253 skip(XmlPullParser parser)254 private static void skip(XmlPullParser parser) throws XmlPullParserException, IOException { 255 int depth = 1; 256 while (depth > 0) { 257 switch (parser.next()) { 258 case XmlPullParser.START_TAG: 259 depth++; 260 break; 261 case XmlPullParser.END_TAG: 262 depth--; 263 break; 264 } 265 } 266 } 267 } 268