1 /*
2  * Copyright (C) 2010 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 
17 package android.graphics;
18 
19 import com.android.SdkConstants;
20 import com.android.ide.common.rendering.api.LayoutLog;
21 import com.android.ide.common.rendering.api.ResourceNamespace;
22 import com.android.layoutlib.bridge.Bridge;
23 import com.android.layoutlib.bridge.android.BridgeContext;
24 import com.android.layoutlib.bridge.android.BridgeXmlBlockParser;
25 import com.android.layoutlib.bridge.android.RenderParamsFlags;
26 import com.android.layoutlib.bridge.impl.DelegateManager;
27 import com.android.layoutlib.bridge.impl.RenderAction;
28 import com.android.tools.layoutlib.annotations.LayoutlibDelegate;
29 
30 import org.xmlpull.v1.XmlPullParser;
31 import org.xmlpull.v1.XmlPullParserException;
32 
33 import android.annotation.NonNull;
34 import android.annotation.Nullable;
35 import android.content.res.FontResourcesParser;
36 import android.graphics.FontFamily_Delegate.FontVariant;
37 import android.graphics.fonts.FontFamily_Builder_Delegate;
38 import android.graphics.fonts.FontVariationAxis;
39 
40 import java.awt.Font;
41 import java.io.IOException;
42 import java.nio.file.Files;
43 import java.nio.file.Paths;
44 import java.util.ArrayList;
45 import java.util.HashMap;
46 import java.util.Iterator;
47 import java.util.List;
48 import java.util.Map;
49 import java.util.Spliterator;
50 import java.util.Spliterators;
51 
52 import libcore.util.NativeAllocationRegistry_Delegate;
53 
54 /**
55  * Delegate implementing the native methods of android.graphics.Typeface
56  * <p>
57  * Through the layoutlib_create tool, the original native methods of Typeface have been replaced by
58  * calls to methods of the same name in this delegate class.
59  * <p>
60  * This class behaves like the original native implementation, but in Java, keeping previously
61  * native data into its own objects and mapping them to int that are sent back and forth between it
62  * and the original Typeface class.
63  *
64  * @see DelegateManager
65  */
66 public final class Typeface_Delegate {
67 
68     public static final String SYSTEM_FONTS = "/system/fonts/";
69 
70     public static final Map<String, FontFamily_Delegate[]> sGenericNativeFamilies = new HashMap<>();
71 
72     // ---- delegate manager ----
73     private static final DelegateManager<Typeface_Delegate> sManager =
74             new DelegateManager<>(Typeface_Delegate.class);
75     private static long sFinalizer = -1;
76 
77     // ---- delegate data ----
78     private static long sDefaultTypeface;
79     @NonNull
80     private final FontFamily_Delegate[] mFontFamilies;  // the reference to FontFamily_Delegate.
81     @NonNull
82     private final FontFamily_Builder_Delegate[] mFontFamilyBuilders;  // the reference to
83     // FontFamily_Builder_Delegate.
84     /** @see Font#getStyle() */
85     private final int mStyle;
86     private final int mWeight;
87 
88 
89     // ---- Public Helper methods ----
90 
Typeface_Delegate(@onNull FontFamily_Delegate[] fontFamilies, @NonNull FontFamily_Builder_Delegate[] fontFamilyBuilders, int style, int weight)91     private Typeface_Delegate(@NonNull FontFamily_Delegate[] fontFamilies,
92             @NonNull FontFamily_Builder_Delegate[] fontFamilyBuilders, int style,
93             int weight) {
94         mFontFamilies = fontFamilies;
95         mFontFamilyBuilders = fontFamilyBuilders;
96         mStyle = style;
97         mWeight = weight;
98     }
99 
getDelegate(long nativeTypeface)100     public static Typeface_Delegate getDelegate(long nativeTypeface) {
101         return sManager.getDelegate(nativeTypeface);
102     }
103 
104     // ---- native methods ----
105 
106     @LayoutlibDelegate
nativeCreateFromTypeface(long native_instance, int style)107     /*package*/ static synchronized long nativeCreateFromTypeface(long native_instance, int style) {
108         Typeface_Delegate delegate = sManager.getDelegate(native_instance);
109         if (delegate == null) {
110             delegate = sManager.getDelegate(sDefaultTypeface);
111         }
112         if (delegate == null) {
113             return 0;
114         }
115 
116         return sManager.addNewDelegate(
117                 new Typeface_Delegate(delegate.mFontFamilies, delegate.mFontFamilyBuilders, style,
118                         delegate.mWeight));
119     }
120 
121     @LayoutlibDelegate
nativeCreateFromTypefaceWithExactStyle(long native_instance, int weight, boolean italic)122     /*package*/ static long nativeCreateFromTypefaceWithExactStyle(long native_instance, int weight,
123             boolean italic) {
124         Typeface_Delegate delegate = sManager.getDelegate(native_instance);
125         if (delegate == null) {
126             delegate = sManager.getDelegate(sDefaultTypeface);
127         }
128         if (delegate == null) {
129             return 0;
130         }
131 
132         int style = weight >= 600 ? (italic ? Typeface.BOLD_ITALIC : Typeface.BOLD) :
133                 (italic ? Typeface.ITALIC : Typeface.NORMAL);
134         return sManager.addNewDelegate(
135                 new Typeface_Delegate(delegate.mFontFamilies, delegate.mFontFamilyBuilders, style,
136                         weight));
137     }
138 
139     @LayoutlibDelegate
nativeCreateFromTypefaceWithVariation(long native_instance, List<FontVariationAxis> axes)140     /*package*/ static synchronized long nativeCreateFromTypefaceWithVariation(long native_instance,
141             List<FontVariationAxis> axes) {
142         long newInstance = nativeCreateFromTypeface(native_instance, 0);
143 
144         if (newInstance != 0) {
145             Bridge.getLog().fidelityWarning(LayoutLog.TAG_UNSUPPORTED,
146                     "nativeCreateFromTypefaceWithVariation is not supported", null, null, null);
147         }
148         return newInstance;
149     }
150 
151     @LayoutlibDelegate
nativeGetSupportedAxes(long native_instance)152     /*package*/ static synchronized int[] nativeGetSupportedAxes(long native_instance) {
153         // nativeCreateFromTypefaceWithVariation is not supported so we do not keep the axes
154         return null;
155     }
156 
157     @LayoutlibDelegate
nativeCreateWeightAlias(long native_instance, int weight)158     /*package*/ static long nativeCreateWeightAlias(long native_instance, int weight) {
159         Typeface_Delegate delegate = sManager.getDelegate(native_instance);
160         if (delegate == null) {
161             delegate = sManager.getDelegate(sDefaultTypeface);
162         }
163         if (delegate == null) {
164             return 0;
165         }
166         Typeface_Delegate weightAlias =
167                 new Typeface_Delegate(delegate.mFontFamilies, delegate.mFontFamilyBuilders,
168                         delegate.mStyle,
169                         weight);
170         return sManager.addNewDelegate(weightAlias);
171     }
172 
173     @LayoutlibDelegate
nativeCreateFromArray(long[] familyArray, int weight, int italic)174     /*package*/ static synchronized long nativeCreateFromArray(long[] familyArray, int weight,
175             int italic) {
176         List<FontFamily_Delegate> fontFamilies = new ArrayList<>();
177         List<FontFamily_Builder_Delegate> fontFamilyBuilders = new ArrayList<>();
178         for (long aFamilyArray : familyArray) {
179             try {
180                 fontFamilies.add(FontFamily_Delegate.getDelegate(aFamilyArray));
181             } catch (ClassCastException e) {
182                 fontFamilyBuilders.add(FontFamily_Builder_Delegate.getDelegate(aFamilyArray));
183             }
184         }
185         if (weight == Typeface.RESOLVE_BY_FONT_TABLE) {
186             weight = 400;
187         }
188         if (italic == Typeface.RESOLVE_BY_FONT_TABLE) {
189             italic = 0;
190         }
191         int style = weight >= 600 ? (italic == 1 ? Typeface.BOLD_ITALIC : Typeface.BOLD) :
192                 (italic == 1 ? Typeface.ITALIC : Typeface.NORMAL);
193         Typeface_Delegate delegate =
194                 new Typeface_Delegate(fontFamilies.toArray(new FontFamily_Delegate[0]),
195                 fontFamilyBuilders.toArray(new FontFamily_Builder_Delegate[0]),
196                 style, weight);
197         return sManager.addNewDelegate(delegate);
198     }
199 
200     @LayoutlibDelegate
nativeGetReleaseFunc()201     /*package*/ static long nativeGetReleaseFunc() {
202         synchronized (Typeface_Delegate.class) {
203             if (sFinalizer == -1) {
204                 sFinalizer = NativeAllocationRegistry_Delegate.createFinalizer(
205                         sManager::removeJavaReferenceFor);
206             }
207         }
208         return sFinalizer;
209     }
210 
211     @LayoutlibDelegate
nativeGetStyle(long native_instance)212     /*package*/ static int nativeGetStyle(long native_instance) {
213         Typeface_Delegate delegate = sManager.getDelegate(native_instance);
214         if (delegate == null) {
215             return 0;
216         }
217 
218         return delegate.mStyle;
219     }
220 
221     @LayoutlibDelegate
nativeSetDefault(long native_instance)222     /*package*/ static void nativeSetDefault(long native_instance) {
223         sDefaultTypeface = native_instance;
224     }
225 
226     @LayoutlibDelegate
nativeGetWeight(long native_instance)227     /*package*/ static int nativeGetWeight(long native_instance) {
228         Typeface_Delegate delegate = sManager.getDelegate(native_instance);
229         if (delegate == null) {
230             return 0;
231         }
232         return delegate.mWeight;
233     }
234 
235     /**
236      * Loads a single font or font family from disk
237      */
238     @Nullable
createFromDisk(@onNull BridgeContext context, @NonNull String path, boolean isFramework)239     public static Typeface createFromDisk(@NonNull BridgeContext context, @NonNull String path,
240             boolean isFramework) {
241         // Check if this is an asset that we've already loaded dynamically
242         Typeface typeface = Typeface.findFromCache(context.getAssets(), path);
243         if (typeface != null) {
244             return typeface;
245         }
246 
247         String lowerCaseValue = path.toLowerCase();
248         if (lowerCaseValue.endsWith(SdkConstants.DOT_XML)) {
249             // create a block parser for the file
250             Boolean psiParserSupport = context.getLayoutlibCallback().getFlag(
251                     RenderParamsFlags.FLAG_KEY_XML_FILE_PARSER_SUPPORT);
252             XmlPullParser parser;
253             if (psiParserSupport != null && psiParserSupport) {
254                 parser = context.getLayoutlibCallback().createXmlParserForPsiFile(path);
255             } else {
256                 parser = context.getLayoutlibCallback().createXmlParserForFile(path);
257             }
258 
259             if (parser != null) {
260                 // TODO(b/156609434): The aapt namespace should not matter for parsing font files?
261                 BridgeXmlBlockParser blockParser =
262                         new BridgeXmlBlockParser(
263                                 parser, context, ResourceNamespace.fromBoolean(isFramework));
264                 try {
265                     FontResourcesParser.FamilyResourceEntry entry =
266                             FontResourcesParser.parse(blockParser, context.getResources());
267                     typeface = Typeface.createFromResources(entry, context.getAssets(), path);
268                 } catch (XmlPullParserException | IOException e) {
269                     Bridge.getLog().error(null, "Failed to parse file " + path, e, null, null /*data
270                     */);
271                 } finally {
272                     blockParser.ensurePopped();
273                 }
274             } else {
275                 Bridge.getLog().error(LayoutLog.TAG_BROKEN,
276                         String.format("File %s does not exist (or is not a file)", path),
277                         null, null /*data*/);
278             }
279         } else {
280             typeface = new Typeface.Builder(context.getAssets(), path, false, 0).build();
281         }
282 
283         return typeface;
284     }
285 
286     @LayoutlibDelegate
create(String familyName, int style)287     /*package*/ static Typeface create(String familyName, int style) {
288         if (familyName != null && Files.exists(Paths.get(familyName))) {
289             // Workaround for b/64137851
290             // Support lib will call this method after failing to create the TypefaceCompat.
291             return Typeface_Delegate.createFromDisk(RenderAction.getCurrentContext(), familyName,
292                     false);
293         }
294         return Typeface.create_Original(familyName, style);
295     }
296 
297     @LayoutlibDelegate
create(Typeface family, int style)298     /*package*/ static Typeface create(Typeface family, int style) {
299         return Typeface.create_Original(family, style);
300     }
301 
302     @LayoutlibDelegate
create(Typeface family, int style, boolean isItalic)303     /*package*/ static Typeface create(Typeface family, int style, boolean isItalic) {
304         return Typeface.create_Original(family, style, isItalic);
305     }
306 
307     @LayoutlibDelegate
nativeRegisterGenericFamily(String str, long nativePtr)308     /*package*/ static void nativeRegisterGenericFamily(String str, long nativePtr) {
309         Typeface_Delegate delegate = sManager.getDelegate(nativePtr);
310         if (delegate == null) {
311             return;
312         }
313         sGenericNativeFamilies.put(str, delegate.mFontFamilies);
314     }
315 
316     // ---- Private delegate/helper methods ----
317 
318     /**
319      * Return an Iterable of fonts that match the style and variant. The list is ordered
320      * according to preference of fonts.
321      * <p>
322      * The Iterator may contain null when the font failed to load. If null is reached when trying to
323      * render with this list of fonts, then a warning should be logged letting the user know that
324      * some font failed to load.
325      *
326      * @param variant The variant preferred. Can only be {@link FontVariant#COMPACT} or {@link
327      * FontVariant#ELEGANT}
328      */
329     @NonNull
getFonts(final FontVariant variant)330     public Iterable<Font> getFonts(final FontVariant variant) {
331         assert variant != FontVariant.NONE;
332 
333         return new FontsIterator(mFontFamilies, mFontFamilyBuilders, variant, mWeight, mStyle);
334     }
335 
336     private static class FontsIterator implements Iterator<Font>, Iterable<Font> {
337         private final FontFamily_Delegate[] fontFamilies;
338         private final FontFamily_Builder_Delegate[] fontFamilyBuilders;
339         private final int weight;
340         private final boolean isItalic;
341         private final FontVariant variant;
342 
343         private int index = 0;
344 
FontsIterator(@onNull FontFamily_Delegate[] fontFamilies, @NonNull FontFamily_Builder_Delegate[] fontFamilyBuilders, @NonNull FontVariant variant, int weight, int style)345         private FontsIterator(@NonNull FontFamily_Delegate[] fontFamilies,
346                 @NonNull FontFamily_Builder_Delegate[] fontFamilyBuilders,
347                 @NonNull FontVariant variant, int weight, int style) {
348             // Calculate the required weight based on style and weight of this typeface.
349             int boldExtraWeight =
350                     ((style & Font.BOLD) == 0 ? 0 : FontFamily_Delegate.BOLD_FONT_WEIGHT_DELTA);
351             this.weight = Math.min(Math.max(100, weight + 50 + boldExtraWeight), 1000);
352             this.isItalic = (style & Font.ITALIC) != 0;
353             this.fontFamilies = fontFamilies;
354             this.fontFamilyBuilders = fontFamilyBuilders;
355             this.variant = variant;
356         }
357 
358         @Override
hasNext()359         public boolean hasNext() {
360             return index < (fontFamilies.length + fontFamilyBuilders.length);
361         }
362 
363         @Override
364         @Nullable
next()365         public Font next() {
366             Font font;
367             FontVariant ffdVariant;
368             if (index < fontFamilies.length) {
369                 FontFamily_Delegate ffd = fontFamilies[index++];
370                 if (ffd == null || !ffd.isValid()) {
371                     return null;
372                 }
373                 font = ffd.getFont(weight, isItalic);
374                 ffdVariant = ffd.getVariant();
375             } else {
376                 FontFamily_Builder_Delegate ffd = fontFamilyBuilders[index++ - fontFamilies.length];
377                 if (ffd == null) {
378                     return null;
379                 }
380                 font = ffd.getFont(weight, isItalic);
381                 ffdVariant = ffd.getVariant();
382             }
383 
384             if (font == null) {
385                 // The FontFamily is valid but doesn't contain any matching font. This means
386                 // that the font failed to load. We add null to the list of fonts. Don't throw
387                 // the warning just yet. If this is a non-english font, we don't want to warn
388                 // users who are trying to render only english text.
389                 return null;
390             }
391 
392             if (ffdVariant == FontVariant.NONE || ffdVariant == variant) {
393                 return font;
394             }
395 
396             // We cannot open each font and get locales supported, etc to match the fonts.
397             // As a workaround, we hardcode certain assumptions like Elegant and Compact
398             // always appear in pairs.
399             if (index < fontFamilies.length) {
400                 assert index < fontFamilies.length - 1;
401                 FontFamily_Delegate ffd2 = fontFamilies[index++];
402                 assert ffd2 != null;
403 
404                 return ffd2.getFont(weight, isItalic);
405             } else {
406                 assert index < fontFamilies.length + fontFamilyBuilders.length - 1;
407                 FontFamily_Builder_Delegate ffd2 = fontFamilyBuilders[index++ - fontFamilies.length];
408                 assert ffd2 != null;
409 
410                 return ffd2.getFont(weight, isItalic);
411             }
412         }
413 
414         @NonNull
415         @Override
416         public Iterator<Font> iterator() {
417             return this;
418         }
419 
420         @Override
421         public Spliterator<Font> spliterator() {
422             return Spliterators.spliterator(iterator(),
423                     fontFamilies.length + fontFamilyBuilders.length,
424                     Spliterator.IMMUTABLE | Spliterator.SIZED);
425         }
426     }
427 }
428