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.cts.mockime;
18 
19 import android.os.Bundle;
20 import android.os.PersistableBundle;
21 
22 import androidx.annotation.ColorInt;
23 import androidx.annotation.NonNull;
24 import androidx.annotation.Nullable;
25 
26 /**
27  * An immutable data store to control the behavior of {@link MockIme}.
28  */
29 public class ImeSettings {
30 
31     @NonNull
32     private final String mClientPackageName;
33 
34     @NonNull
35     private final String mEventCallbackActionName;
36 
37     private static final String EVENT_CALLBACK_INTENT_ACTION_KEY = "eventCallbackActionName";
38     private static final String DATA_KEY = "data";
39 
40     private static final String BACKGROUND_COLOR_KEY = "BackgroundColor";
41     private static final String NAVIGATION_BAR_COLOR_KEY = "NavigationBarColor";
42     private static final String INPUT_VIEW_HEIGHT =
43             "InputViewHeightWithoutSystemWindowInset";
44     private static final String DRAWS_BEHIND_NAV_BAR = "drawsBehindNavBar";
45     private static final String WINDOW_FLAGS = "WindowFlags";
46     private static final String WINDOW_FLAGS_MASK = "WindowFlagsMask";
47     private static final String FULLSCREEN_MODE_ALLOWED = "FullscreenModeAllowed";
48     private static final String INPUT_VIEW_SYSTEM_UI_VISIBILITY = "InputViewSystemUiVisibility";
49     private static final String WATERMARK_ENABLED = "WatermarkEnabled";
50     private static final String HARD_KEYBOARD_CONFIGURATION_BEHAVIOR_ALLOWED =
51             "HardKeyboardConfigurationBehaviorAllowed";
52     private static final String INLINE_SUGGESTIONS_ENABLED = "InlineSuggestionsEnabled";
53     private static final String INLINE_SUGGESTION_VIEW_CONTENT_DESC =
54             "InlineSuggestionViewContentDesc";
55     private static final String STRICT_MODE_ENABLED = "StrictModeEnabled";
56     private static final String VERIFY_CONTEXT_APIS_IN_ON_CREATE = "VerifyContextApisInOnCreate";
57 
58     @NonNull
59     private final PersistableBundle mBundle;
60 
ImeSettings(@onNull String clientPackageName, @NonNull Bundle bundle)61     ImeSettings(@NonNull String clientPackageName, @NonNull Bundle bundle) {
62         mClientPackageName = clientPackageName;
63         mEventCallbackActionName = bundle.getString(EVENT_CALLBACK_INTENT_ACTION_KEY);
64         mBundle = bundle.getParcelable(DATA_KEY);
65     }
66 
67     @Nullable
getEventCallbackActionName()68     String getEventCallbackActionName() {
69         return mEventCallbackActionName;
70     }
71 
72     @NonNull
getClientPackageName()73     String getClientPackageName() {
74         return mClientPackageName;
75     }
76 
fullscreenModeAllowed(boolean defaultValue)77     public boolean fullscreenModeAllowed(boolean defaultValue) {
78         return mBundle.getBoolean(FULLSCREEN_MODE_ALLOWED, defaultValue);
79     }
80 
81     @ColorInt
getBackgroundColor(@olorInt int defaultColor)82     public int getBackgroundColor(@ColorInt int defaultColor) {
83         return mBundle.getInt(BACKGROUND_COLOR_KEY, defaultColor);
84     }
85 
hasNavigationBarColor()86     public boolean hasNavigationBarColor() {
87         return mBundle.keySet().contains(NAVIGATION_BAR_COLOR_KEY);
88     }
89 
90     @ColorInt
getNavigationBarColor()91     public int getNavigationBarColor() {
92         return mBundle.getInt(NAVIGATION_BAR_COLOR_KEY);
93     }
94 
getInputViewHeight(int defaultHeight)95     public int getInputViewHeight(int defaultHeight) {
96         return mBundle.getInt(INPUT_VIEW_HEIGHT, defaultHeight);
97     }
98 
getDrawsBehindNavBar()99     public boolean getDrawsBehindNavBar() {
100         return mBundle.getBoolean(DRAWS_BEHIND_NAV_BAR, false);
101     }
102 
getWindowFlags(int defaultFlags)103     public int getWindowFlags(int defaultFlags) {
104         return mBundle.getInt(WINDOW_FLAGS, defaultFlags);
105     }
106 
getWindowFlagsMask(int defaultFlags)107     public int getWindowFlagsMask(int defaultFlags) {
108         return mBundle.getInt(WINDOW_FLAGS_MASK, defaultFlags);
109     }
110 
getInputViewSystemUiVisibility(int defaultFlags)111     public int getInputViewSystemUiVisibility(int defaultFlags) {
112         return mBundle.getInt(INPUT_VIEW_SYSTEM_UI_VISIBILITY, defaultFlags);
113     }
114 
isWatermarkEnabled(boolean defaultValue)115     public boolean isWatermarkEnabled(boolean defaultValue) {
116         return mBundle.getBoolean(WATERMARK_ENABLED, defaultValue);
117     }
118 
getHardKeyboardConfigurationBehaviorAllowed(boolean defaultValue)119     public boolean getHardKeyboardConfigurationBehaviorAllowed(boolean defaultValue) {
120         return mBundle.getBoolean(HARD_KEYBOARD_CONFIGURATION_BEHAVIOR_ALLOWED, defaultValue);
121     }
122 
getInlineSuggestionsEnabled()123     public boolean getInlineSuggestionsEnabled() {
124         return mBundle.getBoolean(INLINE_SUGGESTIONS_ENABLED);
125     }
126 
127     @Nullable
getInlineSuggestionViewContentDesc(@ullable String defaultValue)128     public String getInlineSuggestionViewContentDesc(@Nullable String defaultValue) {
129         return mBundle.getString(INLINE_SUGGESTION_VIEW_CONTENT_DESC, defaultValue);
130     }
131 
isStrictModeEnabled()132     public boolean isStrictModeEnabled() {
133         return mBundle.getBoolean(STRICT_MODE_ENABLED, false);
134     }
135 
isVerifyContextApisInOnCreate()136     public boolean isVerifyContextApisInOnCreate() {
137         return mBundle.getBoolean(VERIFY_CONTEXT_APIS_IN_ON_CREATE, false);
138     }
139 
serializeToBundle(@onNull String eventCallbackActionName, @Nullable Builder builder)140     static Bundle serializeToBundle(@NonNull String eventCallbackActionName,
141             @Nullable Builder builder) {
142         final Bundle result = new Bundle();
143         result.putString(EVENT_CALLBACK_INTENT_ACTION_KEY, eventCallbackActionName);
144         result.putParcelable(DATA_KEY, builder != null ? builder.mBundle : PersistableBundle.EMPTY);
145         return result;
146     }
147 
148     /**
149      * The builder class for {@link ImeSettings}.
150      */
151     public static final class Builder {
152         private final PersistableBundle mBundle = new PersistableBundle();
153 
154         /**
155          * Controls whether fullscreen mode is allowed or not.
156          *
157          * <p>By default, fullscreen mode is not allowed in {@link MockIme}.</p>
158          *
159          * @param allowed {@code true} if fullscreen mode is allowed
160          * @see MockIme#onEvaluateFullscreenMode()
161          */
setFullscreenModeAllowed(boolean allowed)162         public Builder setFullscreenModeAllowed(boolean allowed) {
163             mBundle.putBoolean(FULLSCREEN_MODE_ALLOWED, allowed);
164             return this;
165         }
166 
167         /**
168          * Sets the background color of the {@link MockIme}.
169          * @param color background color to be used
170          */
setBackgroundColor(@olorInt int color)171         public Builder setBackgroundColor(@ColorInt int color) {
172             mBundle.putInt(BACKGROUND_COLOR_KEY, color);
173             return this;
174         }
175 
176         /**
177          * Sets the color to be passed to {@link android.view.Window#setNavigationBarColor(int)}.
178          *
179          * @param color color to be passed to {@link android.view.Window#setNavigationBarColor(int)}
180          * @see android.view.View
181          */
setNavigationBarColor(@olorInt int color)182         public Builder setNavigationBarColor(@ColorInt int color) {
183             mBundle.putInt(NAVIGATION_BAR_COLOR_KEY, color);
184             return this;
185         }
186 
187         /**
188          * Sets the input view height measured from the bottom of the screen.
189          *
190          * @param height height of the soft input view. This includes the system window inset such
191          *               as navigation bar.
192          */
setInputViewHeight(int height)193         public Builder setInputViewHeight(int height) {
194             mBundle.putInt(INPUT_VIEW_HEIGHT, height);
195             return this;
196         }
197 
198         /**
199          * Sets whether IME draws behind navigation bar.
200          */
setDrawsBehindNavBar(boolean drawsBehindNavBar)201         public Builder setDrawsBehindNavBar(boolean drawsBehindNavBar) {
202             mBundle.putBoolean(DRAWS_BEHIND_NAV_BAR, drawsBehindNavBar);
203             return this;
204         }
205 
206         /**
207          * Sets window flags to be specified to {@link android.view.Window#setFlags(int, int)} of
208          * the main {@link MockIme} window.
209          *
210          * <p>When {@link android.view.WindowManager.LayoutParams#FLAG_LAYOUT_IN_OVERSCAN} is set,
211          * {@link MockIme} tries to render the navigation bar by itself.</p>
212          *
213          * @param flags flags to be specified
214          * @param flagsMask mask bits that specify what bits need to be cleared before setting
215          *                  {@code flags}
216          * @see android.view.WindowManager
217          */
setWindowFlags(int flags, int flagsMask)218         public Builder setWindowFlags(int flags, int flagsMask) {
219             mBundle.putInt(WINDOW_FLAGS, flags);
220             mBundle.putInt(WINDOW_FLAGS_MASK, flagsMask);
221             return this;
222         }
223 
224         /**
225          * Sets flags to be specified to {@link android.view.View#setSystemUiVisibility(int)} of
226          * the main soft input view (the returned view from {@link MockIme#onCreateInputView()}).
227          *
228          * @param visibilityFlags flags to be specified
229          * @see android.view.View
230          */
setInputViewSystemUiVisibility(int visibilityFlags)231         public Builder setInputViewSystemUiVisibility(int visibilityFlags) {
232             mBundle.putInt(INPUT_VIEW_SYSTEM_UI_VISIBILITY, visibilityFlags);
233             return this;
234         }
235 
236         /**
237          * Sets whether a unique watermark image needs to be shown on the software keyboard or not.
238          *
239          * <p>This needs to be enabled to use</p>
240          *
241          * @param enabled {@code true} when such a watermark image is requested.
242          */
setWatermarkEnabled(boolean enabled)243         public Builder setWatermarkEnabled(boolean enabled) {
244             mBundle.putBoolean(WATERMARK_ENABLED, enabled);
245             return this;
246         }
247 
248         /**
249          * Controls whether {@link MockIme} is allowed to change the behavior based on
250          * {@link android.content.res.Configuration#keyboard} and
251          * {@link android.content.res.Configuration#hardKeyboardHidden}.
252          *
253          * <p>Methods in {@link android.inputmethodservice.InputMethodService} such as
254          * {@link android.inputmethodservice.InputMethodService#onEvaluateInputViewShown()} and
255          * {@link android.inputmethodservice.InputMethodService#onShowInputRequested(int, boolean)}
256          * change their behaviors when a hardware keyboard is attached.  This is confusing when
257          * writing tests so by default {@link MockIme} tries to cancel those behaviors.  This
258          * settings re-enables such a behavior.</p>
259          *
260          * @param allowed {@code true} when {@link MockIme} is allowed to change the behavior when
261          *                a hardware keyboard is attached
262          *
263          * @see android.inputmethodservice.InputMethodService#onEvaluateInputViewShown()
264          * @see android.inputmethodservice.InputMethodService#onShowInputRequested(int, boolean)
265          */
setHardKeyboardConfigurationBehaviorAllowed(boolean allowed)266         public Builder setHardKeyboardConfigurationBehaviorAllowed(boolean allowed) {
267             mBundle.putBoolean(HARD_KEYBOARD_CONFIGURATION_BEHAVIOR_ALLOWED, allowed);
268             return this;
269         }
270 
271         /**
272          * Controls whether inline suggestions are enabled for {@link MockIme}. If enabled, a
273          * suggestion strip will be rendered at the top of the keyboard.
274          *
275          * @param enabled {@code true} when {@link MockIme} is enabled to show inline suggestions.
276          */
setInlineSuggestionsEnabled(boolean enabled)277         public Builder setInlineSuggestionsEnabled(boolean enabled) {
278             mBundle.putBoolean(INLINE_SUGGESTIONS_ENABLED, enabled);
279             return this;
280         }
281 
282         /**
283          * Controls whether inline suggestions are enabled for {@link MockIme}. If enabled, a
284          * suggestion strip will be rendered at the top of the keyboard.
285          *
286          * @param contentDesc content description to be set to the inline suggestion View.
287          */
setInlineSuggestionViewContentDesc(@onNull String contentDesc)288         public Builder setInlineSuggestionViewContentDesc(@NonNull String contentDesc) {
289             mBundle.putString(INLINE_SUGGESTION_VIEW_CONTENT_DESC, contentDesc);
290             return this;
291         }
292 
293         /** Sets whether to enable {@link android.os.StrictMode} or not. */
setStrictModeEnabled(boolean enabled)294         public Builder setStrictModeEnabled(boolean enabled) {
295             mBundle.putBoolean(STRICT_MODE_ENABLED, enabled);
296             return this;
297         }
298 
299         /**
300          * Sets whether to verify below {@link android.content.Context} APIs or not:
301          * <ul>
302          *     <li>{@link android.inputmethodservice.InputMethodService#getDisplay}</li>
303          *     <li>{@link android.inputmethodservice.InputMethodService#isUiContext}</li>
304          * </ul>
305          */
setVerifyUiContextApisInOnCreate(boolean enabled)306         public Builder setVerifyUiContextApisInOnCreate(boolean enabled) {
307             mBundle.putBoolean(VERIFY_CONTEXT_APIS_IN_ON_CREATE, enabled);
308             return this;
309         }
310     }
311 }
312