blob: ec26f5849c76acbf39e4267c5811ac1dd3747a87 [file] [log] [blame]
Winson Chungde34aa42015-05-07 18:21:28 -07001/*
2 * Copyright (C) 2015 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 */
Winson97b0d082015-08-13 15:18:25 -070016package com.android.launcher3;
Winson Chungde34aa42015-05-07 18:21:28 -070017
Anushree Ganjama4616602022-07-25 22:20:18 +000018import static com.android.launcher3.logging.KeyboardStateManager.KeyboardState.SHOW;
Sunny Goyal59969372021-05-06 12:11:44 -070019
Winson Chungde34aa42015-05-07 18:21:28 -070020import android.content.Context;
Luca Zuccarini0a3d67f2022-11-17 13:04:59 +000021import android.graphics.Rect;
Sunny Goyal326403e2017-10-02 12:45:10 -070022import android.text.TextUtils;
Winson Chungde34aa42015-05-07 18:21:28 -070023import android.util.AttributeSet;
Luca Zuccarini95bc2ce2023-08-04 13:18:16 +000024import android.util.Log;
Winson4e7a1012015-08-13 16:47:55 -070025import android.view.DragEvent;
Winson Chungde34aa42015-05-07 18:21:28 -070026import android.view.KeyEvent;
Hyunyoung Songc2fe1142016-09-09 15:02:20 -070027import android.view.inputmethod.InputMethodManager;
Winson Chungde34aa42015-05-07 18:21:28 -070028import android.widget.EditText;
29
Sunny Goyal59969372021-05-06 12:11:44 -070030import com.android.launcher3.views.ActivityContext;
Sunny Goyal326403e2017-10-02 12:45:10 -070031
Luca Zuccarini0a3d67f2022-11-17 13:04:59 +000032import java.util.HashSet;
33import java.util.Set;
34
Winson Chungde34aa42015-05-07 18:21:28 -070035
36/**
Winson97b0d082015-08-13 15:18:25 -070037 * The edit text that reports back when the back key has been pressed.
Hyunyoung Song5fcd8f92019-11-06 21:34:36 -080038 * Note: AppCompatEditText doesn't fully support #displayCompletions and #onCommitCompletion
Winson Chungde34aa42015-05-07 18:21:28 -070039 */
Winson97b0d082015-08-13 15:18:25 -070040public class ExtendedEditText extends EditText {
Luca Zuccarini95bc2ce2023-08-04 13:18:16 +000041 private static final String TAG = "ExtendedEditText";
42
Luca Zuccarini0a3d67f2022-11-17 13:04:59 +000043 private final Set<OnFocusChangeListener> mOnFocusChangeListeners = new HashSet<>();
44
Jon Miranda54d4e642017-02-24 10:00:14 -080045 private boolean mForceDisableSuggestions = false;
Hyunyoung Songc2fe1142016-09-09 15:02:20 -070046
Winson Chungde34aa42015-05-07 18:21:28 -070047 /**
48 * Implemented by listeners of the back key.
49 */
50 public interface OnBackKeyListener {
Hyunyoung Song54d1f882020-01-10 23:37:16 -080051 boolean onBackKey();
Winson Chungde34aa42015-05-07 18:21:28 -070052 }
53
54 private OnBackKeyListener mBackKeyListener;
55
Winson97b0d082015-08-13 15:18:25 -070056 public ExtendedEditText(Context context) {
Hyunyoung Song3f9d6472016-09-20 12:37:41 -070057 // ctor chaining breaks the touch handling
58 super(context);
Winson Chungde34aa42015-05-07 18:21:28 -070059 }
60
Winson97b0d082015-08-13 15:18:25 -070061 public ExtendedEditText(Context context, AttributeSet attrs) {
Hyunyoung Song3f9d6472016-09-20 12:37:41 -070062 // ctor chaining breaks the touch handling
63 super(context, attrs);
Winson Chungde34aa42015-05-07 18:21:28 -070064 }
65
Winson97b0d082015-08-13 15:18:25 -070066 public ExtendedEditText(Context context, AttributeSet attrs, int defStyleAttr) {
Winson Chungde34aa42015-05-07 18:21:28 -070067 super(context, attrs, defStyleAttr);
68 }
69
70 public void setOnBackKeyListener(OnBackKeyListener listener) {
71 mBackKeyListener = listener;
72 }
73
74 @Override
75 public boolean onKeyPreIme(int keyCode, KeyEvent event) {
76 // If this is a back key, propagate the key back to the listener
Andy Wickham336c6662023-08-25 16:56:30 -070077 if (keyCode == KeyEvent.KEYCODE_BACK && event.getAction() == KeyEvent.ACTION_UP
78 && mBackKeyListener != null) {
79 return mBackKeyListener.onBackKey();
Winson Chungde34aa42015-05-07 18:21:28 -070080 }
81 return super.onKeyPreIme(keyCode, event);
82 }
Winson4e7a1012015-08-13 16:47:55 -070083
84 @Override
85 public boolean onDragEvent(DragEvent event) {
86 // We don't want this view to interfere with Launcher own drag and drop.
87 return false;
88 }
Hyunyoung Songc2fe1142016-09-09 15:02:20 -070089
Luca Zuccarini95bc2ce2023-08-04 13:18:16 +000090 /**
91 * Synchronously shows the soft input method.
92 *
Luca Zuccarini6ab61062023-11-13 17:58:19 +000093 * @return true if the keyboard is shown correctly and focus is given to this view.
Luca Zuccarini95bc2ce2023-08-04 13:18:16 +000094 */
Luca Zuccarini6ab61062023-11-13 17:58:19 +000095 public boolean showKeyboard() {
Anushree Ganjama4616602022-07-25 22:20:18 +000096 onKeyboardShown();
Luca Zuccarini6ab61062023-11-13 17:58:19 +000097 return requestFocus() && showSoftInputInternal();
98 }
99
100 /**
101 * Requests the framework to show the keyboard in order to ensure that an already registered
102 * controlled keyboard animation is triggered correctly.
103 * Must NEVER be called in any other case than to trigger a pre-registered controlled animation.
104 */
105 public void requestShowKeyboardForControlledAnimation() {
106 // We don't log the keyboard state, as that must happen only after the controlled animation
107 // has completed.
108 // We also must not request focus, as this triggers unwanted side effects.
109 showSoftInputInternal();
Hyunyoung Songc2fe1142016-09-09 15:02:20 -0700110 }
111
Mehdi Alizadeh25b4dfe2018-06-05 16:22:35 -0700112 public void hideKeyboard() {
Pat Manningb53d8d62023-10-25 10:27:40 +0100113 hideKeyboard(/* clearFocus= */ true);
114 }
115
116 public void hideKeyboard(boolean clearFocus) {
Sunny Goyal8958a702022-09-09 15:54:10 -0700117 ActivityContext.lookupContext(getContext()).hideKeyboard();
Pat Manningb53d8d62023-10-25 10:27:40 +0100118 if (clearFocus) {
119 clearFocus();
120 }
Mehdi Alizadeh25b4dfe2018-06-05 16:22:35 -0700121 }
122
Anushree Ganjama4616602022-07-25 22:20:18 +0000123 protected void onKeyboardShown() {
124 ActivityContext.lookupContext(getContext()).getStatsLogManager()
125 .keyboardStateManager().setKeyboardState(SHOW);
126 }
127
Luca Zuccarini95bc2ce2023-08-04 13:18:16 +0000128 private boolean showSoftInputInternal() {
129 boolean result = false;
130 InputMethodManager imm = getContext().getSystemService(InputMethodManager.class);
131 if (imm != null) {
132 result = imm.showSoftInput(this, InputMethodManager.SHOW_IMPLICIT);
133 } else {
134 Log.w(TAG, "Failed to retrieve InputMethodManager from the system.");
135 }
136 return result;
Hyunyoung Songc2fe1142016-09-09 15:02:20 -0700137 }
Sunny Goyal740ac7f2016-09-28 16:47:32 -0700138
139 public void dispatchBackKey() {
Mehdi Alizadeh25b4dfe2018-06-05 16:22:35 -0700140 hideKeyboard();
Sunny Goyal740ac7f2016-09-28 16:47:32 -0700141 if (mBackKeyListener != null) {
142 mBackKeyListener.onBackKey();
143 }
144 }
Jon Miranda54d4e642017-02-24 10:00:14 -0800145
146 /**
147 * Set to true when you want isSuggestionsEnabled to return false.
148 * Use this to disable the red underlines that appear under typos when suggestions is enabled.
149 */
150 public void forceDisableSuggestions(boolean forceDisableSuggestions) {
151 mForceDisableSuggestions = forceDisableSuggestions;
152 }
153
154 @Override
155 public boolean isSuggestionsEnabled() {
156 return !mForceDisableSuggestions && super.isSuggestionsEnabled();
157 }
Sunny Goyal326403e2017-10-02 12:45:10 -0700158
159 public void reset() {
160 if (!TextUtils.isEmpty(getText())) {
161 setText("");
Hyunyoung Songa907a992021-03-10 10:18:03 -0800162 }
Sunny Goyal326403e2017-10-02 12:45:10 -0700163 }
Luca Zuccarini0a3d67f2022-11-17 13:04:59 +0000164
Anushree Ganjamc3bf3942023-11-15 13:31:26 -0800165 @Override
166 public void setText(CharSequence text, BufferType type) {
167 super.setText(text, type);
168 // With hardware keyboard, there is a possibility that the user types before edit
169 // text is visible during the transition.
170 // So move the cursor to the end of the text.
171 setSelection(getText().length());
172 }
173
Luca Zuccarini0a3d67f2022-11-17 13:04:59 +0000174 /**
175 * This method should be preferred to {@link #setOnFocusChangeListener(OnFocusChangeListener)},
176 * as it allows for multiple listeners from different sources.
177 */
178 public void addOnFocusChangeListener(OnFocusChangeListener listener) {
179 mOnFocusChangeListeners.add(listener);
180 }
181
182 /**
183 * Removes the given listener from the set of registered focus listeners, or does nothing if it
184 * wasn't registered in the first place.
185 */
186 public void removeOnFocusChangeListener(OnFocusChangeListener listener) {
187 mOnFocusChangeListeners.remove(listener);
188 }
189
190 @Override
191 protected void onFocusChanged(boolean focused, int direction, Rect previouslyFocusedRect) {
192 super.onFocusChanged(focused, direction, previouslyFocusedRect);
193 for (OnFocusChangeListener listener : mOnFocusChangeListeners) {
194 listener.onFocusChange(this, focused);
195 }
196 }
Holly Sun196d55a2022-12-08 14:05:37 -0800197
198 /**
199 * Save the input, suggestion, hint states when it's on focus, and set to unfocused states.
200 */
201 public void saveFocusedStateAndUpdateToUnfocusedState() {}
202
203 /**
204 * Restore to the previous saved focused state.
205 */
206 public void restoreToFocusedState() {}
Winson Chungde34aa42015-05-07 18:21:28 -0700207}