• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2006 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.content.res;
18 
19 import android.content.pm.ApplicationInfo;
20 import android.graphics.Canvas;
21 import android.graphics.PointF;
22 import android.graphics.Rect;
23 import android.graphics.Region;
24 import android.os.Parcel;
25 import android.os.Parcelable;
26 import android.util.DisplayMetrics;
27 import android.view.MotionEvent;
28 import android.view.WindowManager;
29 import android.view.WindowManager.LayoutParams;
30 
31 /**
32  * CompatibilityInfo class keeps the information about compatibility mode that the application is
33  * running under.
34  *
35  *  {@hide}
36  */
37 public class CompatibilityInfo implements Parcelable {
38     /** default compatibility info object for compatible applications */
39     public static final CompatibilityInfo DEFAULT_COMPATIBILITY_INFO = new CompatibilityInfo() {
40     };
41 
42     /**
43      * This is the number of pixels we would like to have along the
44      * short axis of an app that needs to run on a normal size screen.
45      */
46     public static final int DEFAULT_NORMAL_SHORT_DIMENSION = 320;
47 
48     /**
49      * This is the maximum aspect ratio we will allow while keeping
50      * applications in a compatible screen size.
51      */
52     public static final float MAXIMUM_ASPECT_RATIO = (854f/480f);
53 
54     /**
55      *  A compatibility flags
56      */
57     private final int mCompatibilityFlags;
58 
59     /**
60      * A flag mask to tell if the application needs scaling (when mApplicationScale != 1.0f)
61      * {@see compatibilityFlag}
62      */
63     private static final int SCALING_REQUIRED = 1;
64 
65     /**
66      * Application must always run in compatibility mode?
67      */
68     private static final int ALWAYS_NEEDS_COMPAT = 2;
69 
70     /**
71      * Application never should run in compatibility mode?
72      */
73     private static final int NEVER_NEEDS_COMPAT = 4;
74 
75     /**
76      * Set if the application needs to run in screen size compatibility mode.
77      */
78     private static final int NEEDS_SCREEN_COMPAT = 8;
79 
80     /**
81      * The effective screen density we have selected for this application.
82      */
83     public final int applicationDensity;
84 
85     /**
86      * Application's scale.
87      */
88     public final float applicationScale;
89 
90     /**
91      * Application's inverted scale.
92      */
93     public final float applicationInvertedScale;
94 
CompatibilityInfo(ApplicationInfo appInfo, int screenLayout, int sw, boolean forceCompat)95     public CompatibilityInfo(ApplicationInfo appInfo, int screenLayout, int sw,
96             boolean forceCompat) {
97         int compatFlags = 0;
98 
99         if (appInfo.requiresSmallestWidthDp != 0 || appInfo.compatibleWidthLimitDp != 0
100                 || appInfo.largestWidthLimitDp != 0) {
101             // New style screen requirements spec.
102             int required = appInfo.requiresSmallestWidthDp != 0
103                     ? appInfo.requiresSmallestWidthDp
104                     : appInfo.compatibleWidthLimitDp;
105             if (required == 0) {
106                 required = appInfo.largestWidthLimitDp;
107             }
108             int compat = appInfo.compatibleWidthLimitDp != 0
109                     ? appInfo.compatibleWidthLimitDp : required;
110             if (compat < required)  {
111                 compat = required;
112             }
113             int largest = appInfo.largestWidthLimitDp;
114 
115             if (required > DEFAULT_NORMAL_SHORT_DIMENSION) {
116                 // For now -- if they require a size larger than the only
117                 // size we can do in compatibility mode, then don't ever
118                 // allow the app to go in to compat mode.  Trying to run
119                 // it at a smaller size it can handle will make it far more
120                 // broken than running at a larger size than it wants or
121                 // thinks it can handle.
122                 compatFlags |= NEVER_NEEDS_COMPAT;
123             } else if (largest != 0 && sw > largest) {
124                 // If the screen size is larger than the largest size the
125                 // app thinks it can work with, then always force it in to
126                 // compatibility mode.
127                 compatFlags |= NEEDS_SCREEN_COMPAT | ALWAYS_NEEDS_COMPAT;
128             } else if (compat >= sw) {
129                 // The screen size is something the app says it was designed
130                 // for, so never do compatibility mode.
131                 compatFlags |= NEVER_NEEDS_COMPAT;
132             } else if (forceCompat) {
133                 // The app may work better with or without compatibility mode.
134                 // Let the user decide.
135                 compatFlags |= NEEDS_SCREEN_COMPAT;
136             }
137 
138             // Modern apps always support densities.
139             applicationDensity = DisplayMetrics.DENSITY_DEVICE;
140             applicationScale = 1.0f;
141             applicationInvertedScale = 1.0f;
142 
143         } else {
144             /**
145              * Has the application said that its UI is expandable?  Based on the
146              * <supports-screen> android:expandible in the manifest.
147              */
148             final int EXPANDABLE = 2;
149 
150             /**
151              * Has the application said that its UI supports large screens?  Based on the
152              * <supports-screen> android:largeScreens in the manifest.
153              */
154             final int LARGE_SCREENS = 8;
155 
156             /**
157              * Has the application said that its UI supports xlarge screens?  Based on the
158              * <supports-screen> android:xlargeScreens in the manifest.
159              */
160             final int XLARGE_SCREENS = 32;
161 
162             int sizeInfo = 0;
163 
164             // We can't rely on the application always setting
165             // FLAG_RESIZEABLE_FOR_SCREENS so will compute it based on various input.
166             boolean anyResizeable = false;
167 
168             if ((appInfo.flags & ApplicationInfo.FLAG_SUPPORTS_LARGE_SCREENS) != 0) {
169                 sizeInfo |= LARGE_SCREENS;
170                 anyResizeable = true;
171                 if (!forceCompat) {
172                     // If we aren't forcing the app into compatibility mode, then
173                     // assume if it supports large screens that we should allow it
174                     // to use the full space of an xlarge screen as well.
175                     sizeInfo |= XLARGE_SCREENS | EXPANDABLE;
176                 }
177             }
178             if ((appInfo.flags & ApplicationInfo.FLAG_SUPPORTS_XLARGE_SCREENS) != 0) {
179                 anyResizeable = true;
180                 if (!forceCompat) {
181                     sizeInfo |= XLARGE_SCREENS | EXPANDABLE;
182                 }
183             }
184             if ((appInfo.flags & ApplicationInfo.FLAG_RESIZEABLE_FOR_SCREENS) != 0) {
185                 anyResizeable = true;
186                 sizeInfo |= EXPANDABLE;
187             }
188 
189             if (forceCompat) {
190                 // If we are forcing compatibility mode, then ignore an app that
191                 // just says it is resizable for screens.  We'll only have it fill
192                 // the screen if it explicitly says it supports the screen size we
193                 // are running in.
194                 sizeInfo &= ~EXPANDABLE;
195             }
196 
197             compatFlags |= NEEDS_SCREEN_COMPAT;
198             switch (screenLayout&Configuration.SCREENLAYOUT_SIZE_MASK) {
199                 case Configuration.SCREENLAYOUT_SIZE_XLARGE:
200                     if ((sizeInfo&XLARGE_SCREENS) != 0) {
201                         compatFlags &= ~NEEDS_SCREEN_COMPAT;
202                     }
203                     if ((appInfo.flags & ApplicationInfo.FLAG_SUPPORTS_XLARGE_SCREENS) != 0) {
204                         compatFlags |= NEVER_NEEDS_COMPAT;
205                     }
206                     break;
207                 case Configuration.SCREENLAYOUT_SIZE_LARGE:
208                     if ((sizeInfo&LARGE_SCREENS) != 0) {
209                         compatFlags &= ~NEEDS_SCREEN_COMPAT;
210                     }
211                     if ((appInfo.flags & ApplicationInfo.FLAG_SUPPORTS_LARGE_SCREENS) != 0) {
212                         compatFlags |= NEVER_NEEDS_COMPAT;
213                     }
214                     break;
215             }
216 
217             if ((screenLayout&Configuration.SCREENLAYOUT_COMPAT_NEEDED) != 0) {
218                 if ((sizeInfo&EXPANDABLE) != 0) {
219                     compatFlags &= ~NEEDS_SCREEN_COMPAT;
220                 } else if (!anyResizeable) {
221                     compatFlags |= ALWAYS_NEEDS_COMPAT;
222                 }
223             } else {
224                 compatFlags &= ~NEEDS_SCREEN_COMPAT;
225                 compatFlags |= NEVER_NEEDS_COMPAT;
226             }
227 
228             if ((appInfo.flags & ApplicationInfo.FLAG_SUPPORTS_SCREEN_DENSITIES) != 0) {
229                 applicationDensity = DisplayMetrics.DENSITY_DEVICE;
230                 applicationScale = 1.0f;
231                 applicationInvertedScale = 1.0f;
232             } else {
233                 applicationDensity = DisplayMetrics.DENSITY_DEFAULT;
234                 applicationScale = DisplayMetrics.DENSITY_DEVICE
235                         / (float) DisplayMetrics.DENSITY_DEFAULT;
236                 applicationInvertedScale = 1.0f / applicationScale;
237                 compatFlags |= SCALING_REQUIRED;
238             }
239         }
240 
241         mCompatibilityFlags = compatFlags;
242     }
243 
CompatibilityInfo(int compFlags, int dens, float scale, float invertedScale)244     private CompatibilityInfo(int compFlags,
245             int dens, float scale, float invertedScale) {
246         mCompatibilityFlags = compFlags;
247         applicationDensity = dens;
248         applicationScale = scale;
249         applicationInvertedScale = invertedScale;
250     }
251 
CompatibilityInfo()252     private CompatibilityInfo() {
253         this(NEVER_NEEDS_COMPAT, DisplayMetrics.DENSITY_DEVICE,
254                 1.0f,
255                 1.0f);
256     }
257 
258     /**
259      * @return true if the scaling is required
260      */
isScalingRequired()261     public boolean isScalingRequired() {
262         return (mCompatibilityFlags&SCALING_REQUIRED) != 0;
263     }
264 
supportsScreen()265     public boolean supportsScreen() {
266         return (mCompatibilityFlags&NEEDS_SCREEN_COMPAT) == 0;
267     }
268 
neverSupportsScreen()269     public boolean neverSupportsScreen() {
270         return (mCompatibilityFlags&ALWAYS_NEEDS_COMPAT) != 0;
271     }
272 
alwaysSupportsScreen()273     public boolean alwaysSupportsScreen() {
274         return (mCompatibilityFlags&NEVER_NEEDS_COMPAT) != 0;
275     }
276 
277     /**
278      * Returns the translator which translates the coordinates in compatibility mode.
279      * @param params the window's parameter
280      */
getTranslator()281     public Translator getTranslator() {
282         return isScalingRequired() ? new Translator() : null;
283     }
284 
285     /**
286      * A helper object to translate the screen and window coordinates back and forth.
287      * @hide
288      */
289     public class Translator {
290         final public float applicationScale;
291         final public float applicationInvertedScale;
292 
293         private Rect mContentInsetsBuffer = null;
294         private Rect mVisibleInsetsBuffer = null;
295         private Region mTouchableAreaBuffer = null;
296 
Translator(float applicationScale, float applicationInvertedScale)297         Translator(float applicationScale, float applicationInvertedScale) {
298             this.applicationScale = applicationScale;
299             this.applicationInvertedScale = applicationInvertedScale;
300         }
301 
Translator()302         Translator() {
303             this(CompatibilityInfo.this.applicationScale,
304                     CompatibilityInfo.this.applicationInvertedScale);
305         }
306 
307         /**
308          * Translate the screen rect to the application frame.
309          */
translateRectInScreenToAppWinFrame(Rect rect)310         public void translateRectInScreenToAppWinFrame(Rect rect) {
311             rect.scale(applicationInvertedScale);
312         }
313 
314         /**
315          * Translate the region in window to screen.
316          */
translateRegionInWindowToScreen(Region transparentRegion)317         public void translateRegionInWindowToScreen(Region transparentRegion) {
318             transparentRegion.scale(applicationScale);
319         }
320 
321         /**
322          * Apply translation to the canvas that is necessary to draw the content.
323          */
translateCanvas(Canvas canvas)324         public void translateCanvas(Canvas canvas) {
325             if (applicationScale == 1.5f) {
326                 /*  When we scale for compatibility, we can put our stretched
327                     bitmaps and ninepatches on exacty 1/2 pixel boundaries,
328                     which can give us inconsistent drawing due to imperfect
329                     float precision in the graphics engine's inverse matrix.
330 
331                     As a work-around, we translate by a tiny amount to avoid
332                     landing on exact pixel centers and boundaries, giving us
333                     the slop we need to draw consistently.
334 
335                     This constant is meant to resolve to 1/255 after it is
336                     scaled by 1.5 (applicationScale). Note, this is just a guess
337                     as to what is small enough not to create its own artifacts,
338                     and big enough to avoid the precision problems. Feel free
339                     to experiment with smaller values as you choose.
340                  */
341                 final float tinyOffset = 2.0f / (3 * 255);
342                 canvas.translate(tinyOffset, tinyOffset);
343             }
344             canvas.scale(applicationScale, applicationScale);
345         }
346 
347         /**
348          * Translate the motion event captured on screen to the application's window.
349          */
translateEventInScreenToAppWindow(MotionEvent event)350         public void translateEventInScreenToAppWindow(MotionEvent event) {
351             event.scale(applicationInvertedScale);
352         }
353 
354         /**
355          * Translate the window's layout parameter, from application's view to
356          * Screen's view.
357          */
translateWindowLayout(WindowManager.LayoutParams params)358         public void translateWindowLayout(WindowManager.LayoutParams params) {
359             params.scale(applicationScale);
360         }
361 
362         /**
363          * Translate a Rect in application's window to screen.
364          */
translateRectInAppWindowToScreen(Rect rect)365         public void translateRectInAppWindowToScreen(Rect rect) {
366             rect.scale(applicationScale);
367         }
368 
369         /**
370          * Translate a Rect in screen coordinates into the app window's coordinates.
371          */
translateRectInScreenToAppWindow(Rect rect)372         public void translateRectInScreenToAppWindow(Rect rect) {
373             rect.scale(applicationInvertedScale);
374         }
375 
376         /**
377          * Translate a Point in screen coordinates into the app window's coordinates.
378          */
translatePointInScreenToAppWindow(PointF point)379         public void translatePointInScreenToAppWindow(PointF point) {
380             final float scale = applicationInvertedScale;
381             if (scale != 1.0f) {
382                 point.x *= scale;
383                 point.y *= scale;
384             }
385         }
386 
387         /**
388          * Translate the location of the sub window.
389          * @param params
390          */
translateLayoutParamsInAppWindowToScreen(LayoutParams params)391         public void translateLayoutParamsInAppWindowToScreen(LayoutParams params) {
392             params.scale(applicationScale);
393         }
394 
395         /**
396          * Translate the content insets in application window to Screen. This uses
397          * the internal buffer for content insets to avoid extra object allocation.
398          */
getTranslatedContentInsets(Rect contentInsets)399         public Rect getTranslatedContentInsets(Rect contentInsets) {
400             if (mContentInsetsBuffer == null) mContentInsetsBuffer = new Rect();
401             mContentInsetsBuffer.set(contentInsets);
402             translateRectInAppWindowToScreen(mContentInsetsBuffer);
403             return mContentInsetsBuffer;
404         }
405 
406         /**
407          * Translate the visible insets in application window to Screen. This uses
408          * the internal buffer for visible insets to avoid extra object allocation.
409          */
getTranslatedVisibleInsets(Rect visibleInsets)410         public Rect getTranslatedVisibleInsets(Rect visibleInsets) {
411             if (mVisibleInsetsBuffer == null) mVisibleInsetsBuffer = new Rect();
412             mVisibleInsetsBuffer.set(visibleInsets);
413             translateRectInAppWindowToScreen(mVisibleInsetsBuffer);
414             return mVisibleInsetsBuffer;
415         }
416 
417         /**
418          * Translate the touchable area in application window to Screen. This uses
419          * the internal buffer for touchable area to avoid extra object allocation.
420          */
getTranslatedTouchableArea(Region touchableArea)421         public Region getTranslatedTouchableArea(Region touchableArea) {
422             if (mTouchableAreaBuffer == null) mTouchableAreaBuffer = new Region();
423             mTouchableAreaBuffer.set(touchableArea);
424             mTouchableAreaBuffer.scale(applicationScale);
425             return mTouchableAreaBuffer;
426         }
427     }
428 
applyToDisplayMetrics(DisplayMetrics inoutDm)429     public void applyToDisplayMetrics(DisplayMetrics inoutDm) {
430         if (!supportsScreen()) {
431             // This is a larger screen device and the app is not
432             // compatible with large screens, so diddle it.
433             CompatibilityInfo.computeCompatibleScaling(inoutDm, inoutDm);
434         } else {
435             inoutDm.widthPixels = inoutDm.noncompatWidthPixels;
436             inoutDm.heightPixels = inoutDm.noncompatHeightPixels;
437         }
438 
439         if (isScalingRequired()) {
440             float invertedRatio = applicationInvertedScale;
441             inoutDm.density = inoutDm.noncompatDensity * invertedRatio;
442             inoutDm.densityDpi = (int)((inoutDm.density*DisplayMetrics.DENSITY_DEFAULT)+.5f);
443             inoutDm.scaledDensity = inoutDm.noncompatScaledDensity * invertedRatio;
444             inoutDm.xdpi = inoutDm.noncompatXdpi * invertedRatio;
445             inoutDm.ydpi = inoutDm.noncompatYdpi * invertedRatio;
446             inoutDm.widthPixels = (int) (inoutDm.widthPixels * invertedRatio + 0.5f);
447             inoutDm.heightPixels = (int) (inoutDm.heightPixels * invertedRatio + 0.5f);
448         }
449     }
450 
applyToConfiguration(Configuration inoutConfig)451     public void applyToConfiguration(Configuration inoutConfig) {
452         if (!supportsScreen()) {
453             // This is a larger screen device and the app is not
454             // compatible with large screens, so we are forcing it to
455             // run as if the screen is normal size.
456             inoutConfig.screenLayout =
457                     (inoutConfig.screenLayout&~Configuration.SCREENLAYOUT_SIZE_MASK)
458                     | Configuration.SCREENLAYOUT_SIZE_NORMAL;
459             inoutConfig.screenWidthDp = inoutConfig.compatScreenWidthDp;
460             inoutConfig.screenHeightDp = inoutConfig.compatScreenHeightDp;
461             inoutConfig.smallestScreenWidthDp = inoutConfig.compatSmallestScreenWidthDp;
462         }
463     }
464 
465     /**
466      * Compute the frame Rect for applications runs under compatibility mode.
467      *
468      * @param dm the display metrics used to compute the frame size.
469      * @param orientation the orientation of the screen.
470      * @param outRect the output parameter which will contain the result.
471      * @return Returns the scaling factor for the window.
472      */
computeCompatibleScaling(DisplayMetrics dm, DisplayMetrics outDm)473     public static float computeCompatibleScaling(DisplayMetrics dm, DisplayMetrics outDm) {
474         final int width = dm.noncompatWidthPixels;
475         final int height = dm.noncompatHeightPixels;
476         int shortSize, longSize;
477         if (width < height) {
478             shortSize = width;
479             longSize = height;
480         } else {
481             shortSize = height;
482             longSize = width;
483         }
484         int newShortSize = (int)(DEFAULT_NORMAL_SHORT_DIMENSION * dm.density + 0.5f);
485         float aspect = ((float)longSize) / shortSize;
486         if (aspect > MAXIMUM_ASPECT_RATIO) {
487             aspect = MAXIMUM_ASPECT_RATIO;
488         }
489         int newLongSize = (int)(newShortSize * aspect + 0.5f);
490         int newWidth, newHeight;
491         if (width < height) {
492             newWidth = newShortSize;
493             newHeight = newLongSize;
494         } else {
495             newWidth = newLongSize;
496             newHeight = newShortSize;
497         }
498 
499         float sw = width/(float)newWidth;
500         float sh = height/(float)newHeight;
501         float scale = sw < sh ? sw : sh;
502         if (scale < 1) {
503             scale = 1;
504         }
505 
506         if (outDm != null) {
507             outDm.widthPixels = newWidth;
508             outDm.heightPixels = newHeight;
509         }
510 
511         return scale;
512     }
513 
514     @Override
515     public boolean equals(Object o) {
516         try {
517             CompatibilityInfo oc = (CompatibilityInfo)o;
518             if (mCompatibilityFlags != oc.mCompatibilityFlags) return false;
519             if (applicationDensity != oc.applicationDensity) return false;
520             if (applicationScale != oc.applicationScale) return false;
521             if (applicationInvertedScale != oc.applicationInvertedScale) return false;
522             return true;
523         } catch (ClassCastException e) {
524             return false;
525         }
526     }
527 
528     @Override
529     public String toString() {
530         StringBuilder sb = new StringBuilder(128);
531         sb.append("{");
532         sb.append(applicationDensity);
533         sb.append("dpi");
534         if (isScalingRequired()) {
535             sb.append(" ");
536             sb.append(applicationScale);
537             sb.append("x");
538         }
539         if (!supportsScreen()) {
540             sb.append(" resizing");
541         }
542         if (neverSupportsScreen()) {
543             sb.append(" never-compat");
544         }
545         if (alwaysSupportsScreen()) {
546             sb.append(" always-compat");
547         }
548         sb.append("}");
549         return sb.toString();
550     }
551 
552     @Override
553     public int hashCode() {
554         int result = 17;
555         result = 31 * result + mCompatibilityFlags;
556         result = 31 * result + applicationDensity;
557         result = 31 * result + Float.floatToIntBits(applicationScale);
558         result = 31 * result + Float.floatToIntBits(applicationInvertedScale);
559         return result;
560     }
561 
562     @Override
563     public int describeContents() {
564         return 0;
565     }
566 
567     @Override
568     public void writeToParcel(Parcel dest, int flags) {
569         dest.writeInt(mCompatibilityFlags);
570         dest.writeInt(applicationDensity);
571         dest.writeFloat(applicationScale);
572         dest.writeFloat(applicationInvertedScale);
573     }
574 
575     public static final Parcelable.Creator<CompatibilityInfo> CREATOR
576             = new Parcelable.Creator<CompatibilityInfo>() {
577         public CompatibilityInfo createFromParcel(Parcel source) {
578             return new CompatibilityInfo(source);
579         }
580 
581         public CompatibilityInfo[] newArray(int size) {
582             return new CompatibilityInfo[size];
583         }
584     };
585 
586     private CompatibilityInfo(Parcel source) {
587         mCompatibilityFlags = source.readInt();
588         applicationDensity = source.readInt();
589         applicationScale = source.readFloat();
590         applicationInvertedScale = source.readFloat();
591     }
592 }
593