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 
17 package com.android.incallui.answer.impl;
18 
19 import android.content.Context;
20 import android.content.res.TypedArray;
21 import android.util.AttributeSet;
22 import android.view.SurfaceView;
23 import android.view.View;
24 import com.android.dialer.common.Assert;
25 
26 /**
27  * A SurfaceView that maintains its aspect ratio to be a desired target value.
28  *
29  * <p>The FixedAspectSurfaceView will not be able to maintain the requested aspect ratio if both the
30  * width and the height are exactly determined by the layout. To avoid this, ensure that either the
31  * height or the width is adjustable by the view; for example, by setting the layout parameters to
32  * be WRAP_CONTENT for the dimension that is best adjusted to maintain the aspect ratio.
33  */
34 public class FixedAspectSurfaceView extends SurfaceView {
35 
36   /** Desired width/height ratio */
37   private float aspectRatio;
38 
39   private final boolean scaleWidth;
40   private final boolean scaleHeight;
41 
FixedAspectSurfaceView(Context context, AttributeSet attrs)42   public FixedAspectSurfaceView(Context context, AttributeSet attrs) {
43     super(context, attrs);
44 
45     // Get initial aspect ratio from custom attributes
46     TypedArray a =
47         context.getTheme().obtainStyledAttributes(attrs, R.styleable.FixedAspectSurfaceView, 0, 0);
48     scaleHeight = a.getBoolean(R.styleable.FixedAspectSurfaceView_scaleHeight, false);
49     scaleWidth = a.getBoolean(R.styleable.FixedAspectSurfaceView_scaleWidth, false);
50     Assert.checkArgument(scaleHeight != scaleWidth, "Must either scale width or height");
51     setAspectRatio(a.getFloat(R.styleable.FixedAspectSurfaceView_aspectRatio, 1.f));
52     a.recycle();
53   }
54 
55   /**
56    * Set the desired aspect ratio for this view.
57    *
58    * @param aspect the desired width/height ratio in the current UI orientation. Must be a positive
59    *     value.
60    */
setAspectRatio(float aspect)61   public void setAspectRatio(float aspect) {
62     Assert.checkArgument(aspect >= 0, "Aspect ratio must be positive");
63     aspectRatio = aspect;
64     requestLayout();
65   }
66 
67   @Override
onMeasure(int widthMeasureSpec, int heightMeasureSpec)68   protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
69     int width = MeasureSpec.getSize(widthMeasureSpec);
70     int height = MeasureSpec.getSize(heightMeasureSpec);
71 
72     // Do the scaling
73     if (scaleWidth) {
74       width = (int) (height * aspectRatio);
75     } else if (scaleHeight) {
76       height = (int) (width / aspectRatio);
77     }
78 
79     // Override width/height if needed for EXACTLY and AT_MOST specs
80     width = View.resolveSizeAndState(width, widthMeasureSpec, 0);
81     height = View.resolveSizeAndState(height, heightMeasureSpec, 0);
82 
83     // Finally set the calculated dimensions
84     setMeasuredDimension(width, height);
85   }
86 }
87