blob: a96b5a0d6ab12047651c70111375934da688db23 [file] [log] [blame]
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001/*
2 * Copyright (C) 2007-2008 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License"); you may not
5 * use this file except in compliance with the License. You may obtain a copy of
6 * 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, WITHOUT
12 * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
13 * License for the specific language governing permissions and limitations under
14 * the License.
15 */
16
17package com.android.internal.widget;
18
19import android.os.Bundle;
20import android.text.Editable;
satokf9f01002011-05-19 21:31:50 +090021import android.text.Spanned;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080022import android.text.method.KeyListener;
satokf9f01002011-05-19 21:31:50 +090023import android.text.style.SuggestionSpan;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080024import android.util.Log;
25import android.view.inputmethod.BaseInputConnection;
26import android.view.inputmethod.CompletionInfo;
Gilles Debunnecf9cf2f2010-12-08 17:43:58 -080027import android.view.inputmethod.CorrectionInfo;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080028import android.view.inputmethod.ExtractedText;
29import android.view.inputmethod.ExtractedTextRequest;
Yohei Yukawaa277db22014-08-21 18:38:44 -070030import android.view.inputmethod.InputConnection;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080031import android.widget.TextView;
32
33public class EditableInputConnection extends BaseInputConnection {
34 private static final boolean DEBUG = false;
35 private static final String TAG = "EditableInputConnection";
36
37 private final TextView mTextView;
38
Gilles Debunnec478c172011-12-19 17:29:24 -080039 // Keeps track of nested begin/end batch edit to ensure this connection always has a
40 // balanced impact on its associated TextView.
41 // A negative value means that this connection has been finished by the InputMethodManager.
42 private int mBatchEditNesting;
43
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080044 public EditableInputConnection(TextView textview) {
Dianne Hackborn51bf0772009-03-24 19:11:41 -070045 super(textview, true);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080046 mTextView = textview;
47 }
48
Gilles Debunnecf9cf2f2010-12-08 17:43:58 -080049 @Override
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080050 public Editable getEditable() {
51 TextView tv = mTextView;
52 if (tv != null) {
53 return tv.getEditableText();
54 }
55 return null;
56 }
Gilles Debunnec478c172011-12-19 17:29:24 -080057
Gilles Debunnecf9cf2f2010-12-08 17:43:58 -080058 @Override
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080059 public boolean beginBatchEdit() {
Gilles Debunnec478c172011-12-19 17:29:24 -080060 synchronized(this) {
61 if (mBatchEditNesting >= 0) {
62 mTextView.beginBatchEdit();
63 mBatchEditNesting++;
64 return true;
65 }
66 }
67 return false;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080068 }
Gilles Debunnec478c172011-12-19 17:29:24 -080069
Gilles Debunnecf9cf2f2010-12-08 17:43:58 -080070 @Override
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080071 public boolean endBatchEdit() {
Gilles Debunnec478c172011-12-19 17:29:24 -080072 synchronized(this) {
73 if (mBatchEditNesting > 0) {
Gilles Debunne9d69ecb2012-02-24 16:07:09 -080074 // When the connection is reset by the InputMethodManager and reportFinish
Gilles Debunnec478c172011-12-19 17:29:24 -080075 // is called, some endBatchEdit calls may still be asynchronously received from the
76 // IME. Do not take these into account, thus ensuring that this IC's final
77 // contribution to mTextView's nested batch edit count is zero.
78 mTextView.endBatchEdit();
79 mBatchEditNesting--;
80 return true;
81 }
82 }
83 return false;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080084 }
Gilles Debunnec478c172011-12-19 17:29:24 -080085
Yohei Yukawaaaa38c92016-03-27 23:46:04 -070086 /**
87 * @hide
88 */
Gilles Debunnecf9cf2f2010-12-08 17:43:58 -080089 @Override
Yohei Yukawaaaa38c92016-03-27 23:46:04 -070090 public void reportFinish() {
Gilles Debunne9d69ecb2012-02-24 16:07:09 -080091 super.reportFinish();
92
93 synchronized(this) {
94 while (mBatchEditNesting > 0) {
95 endBatchEdit();
96 }
97 // Will prevent any further calls to begin or endBatchEdit
98 mBatchEditNesting = -1;
99 }
100 }
101
102 @Override
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800103 public boolean clearMetaKeyStates(int states) {
104 final Editable content = getEditable();
105 if (content == null) return false;
106 KeyListener kl = mTextView.getKeyListener();
107 if (kl != null) {
108 try {
109 kl.clearMetaKeyState(mTextView, content, states);
110 } catch (AbstractMethodError e) {
111 // This is an old listener that doesn't implement the
112 // new method.
113 }
114 }
115 return true;
116 }
Gilles Debunnec478c172011-12-19 17:29:24 -0800117
118 @Override
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800119 public boolean commitCompletion(CompletionInfo text) {
120 if (DEBUG) Log.v(TAG, "commitCompletion " + text);
121 mTextView.beginBatchEdit();
122 mTextView.onCommitCompletion(text);
123 mTextView.endBatchEdit();
124 return true;
125 }
126
Gilles Debunnecf9cf2f2010-12-08 17:43:58 -0800127 /**
128 * Calls the {@link TextView#onCommitCorrection} method of the associated TextView.
129 */
130 @Override
131 public boolean commitCorrection(CorrectionInfo correctionInfo) {
132 if (DEBUG) Log.v(TAG, "commitCorrection" + correctionInfo);
133 mTextView.beginBatchEdit();
134 mTextView.onCommitCorrection(correctionInfo);
135 mTextView.endBatchEdit();
136 return true;
137 }
138
139 @Override
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800140 public boolean performEditorAction(int actionCode) {
141 if (DEBUG) Log.v(TAG, "performEditorAction " + actionCode);
142 mTextView.onEditorAction(actionCode);
143 return true;
144 }
145
Gilles Debunnecf9cf2f2010-12-08 17:43:58 -0800146 @Override
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800147 public boolean performContextMenuAction(int id) {
148 if (DEBUG) Log.v(TAG, "performContextMenuAction " + id);
149 mTextView.beginBatchEdit();
150 mTextView.onTextContextMenuItem(id);
151 mTextView.endBatchEdit();
152 return true;
153 }
154
Gilles Debunnecf9cf2f2010-12-08 17:43:58 -0800155 @Override
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800156 public ExtractedText getExtractedText(ExtractedTextRequest request, int flags) {
157 if (mTextView != null) {
158 ExtractedText et = new ExtractedText();
159 if (mTextView.extractText(request, et)) {
160 if ((flags&GET_EXTRACTED_TEXT_MONITOR) != 0) {
161 mTextView.setExtracting(request);
162 }
163 return et;
164 }
165 }
166 return null;
167 }
Gilles Debunnecf9cf2f2010-12-08 17:43:58 -0800168
169 @Override
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800170 public boolean performPrivateCommand(String action, Bundle data) {
171 mTextView.onPrivateIMECommand(action, data);
172 return true;
173 }
174
175 @Override
176 public boolean commitText(CharSequence text, int newCursorPosition) {
177 if (mTextView == null) {
178 return super.commitText(text, newCursorPosition);
179 }
satokf9f01002011-05-19 21:31:50 +0900180 if (text instanceof Spanned) {
181 Spanned spanned = ((Spanned) text);
182 SuggestionSpan[] spans = spanned.getSpans(0, text.length(), SuggestionSpan.class);
183 mIMM.registerSuggestionSpansForNotification(spans);
184 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800185
Gilles Debunneb7fc63f2011-01-27 15:55:29 -0800186 mTextView.resetErrorChangedFlag();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800187 boolean success = super.commitText(text, newCursorPosition);
Gilles Debunneb7fc63f2011-01-27 15:55:29 -0800188 mTextView.hideErrorIfUnchanged();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800189
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800190 return success;
191 }
Yohei Yukawa0023d0e2014-07-11 04:13:03 +0900192
193 @Override
Yohei Yukawad8636ea2014-09-02 22:03:30 -0700194 public boolean requestCursorUpdates(int cursorUpdateMode) {
Yohei Yukawaa277db22014-08-21 18:38:44 -0700195 if (DEBUG) Log.v(TAG, "requestUpdateCursorAnchorInfo " + cursorUpdateMode);
Yohei Yukawaff328ae2014-07-18 04:22:39 +0900196
Yohei Yukawa72d35fa2014-08-28 23:57:11 -0700197 // It is possible that any other bit is used as a valid flag in a future release.
198 // We should reject the entire request in such a case.
Yohei Yukawad8636ea2014-09-02 22:03:30 -0700199 final int KNOWN_FLAGS_MASK = InputConnection.CURSOR_UPDATE_IMMEDIATE |
200 InputConnection.CURSOR_UPDATE_MONITOR;
Yohei Yukawa72d35fa2014-08-28 23:57:11 -0700201 final int unknownFlags = cursorUpdateMode & ~KNOWN_FLAGS_MASK;
202 if (unknownFlags != 0) {
203 if (DEBUG) {
204 Log.d(TAG, "Rejecting requestUpdateCursorAnchorInfo due to unknown flags." +
205 " cursorUpdateMode=" + cursorUpdateMode +
206 " unknownFlags=" + unknownFlags);
207 }
208 return false;
209 }
210
Yohei Yukawaff328ae2014-07-18 04:22:39 +0900211 if (mIMM == null) {
212 // In this case, TYPE_CURSOR_ANCHOR_INFO is not handled.
Yohei Yukawaa277db22014-08-21 18:38:44 -0700213 // TODO: Return some notification code rather than false to indicate method that
Yohei Yukawaff328ae2014-07-18 04:22:39 +0900214 // CursorAnchorInfo is temporarily unavailable.
Yohei Yukawaa277db22014-08-21 18:38:44 -0700215 return false;
Yohei Yukawaff328ae2014-07-18 04:22:39 +0900216 }
Yohei Yukawaa277db22014-08-21 18:38:44 -0700217 mIMM.setUpdateCursorAnchorInfoMode(cursorUpdateMode);
Yohei Yukawad8636ea2014-09-02 22:03:30 -0700218 if ((cursorUpdateMode & InputConnection.CURSOR_UPDATE_IMMEDIATE) != 0) {
Yohei Yukawaff328ae2014-07-18 04:22:39 +0900219 if (mTextView == null) {
220 // In this case, FLAG_CURSOR_ANCHOR_INFO_IMMEDIATE is silently ignored.
221 // TODO: Return some notification code for the input method that indicates
222 // FLAG_CURSOR_ANCHOR_INFO_IMMEDIATE is ignored.
223 } else if (mTextView.isInLayout()) {
224 // In this case, the view hierarchy is currently undergoing a layout pass.
225 // IMM#updateCursorAnchorInfo is supposed to be called soon after the layout
226 // pass is finished.
227 } else {
228 // This will schedule a layout pass of the view tree, and the layout event
229 // eventually triggers IMM#updateCursorAnchorInfo.
230 mTextView.requestLayout();
231 }
232 }
Yohei Yukawaa277db22014-08-21 18:38:44 -0700233 return true;
Yohei Yukawa0023d0e2014-07-11 04:13:03 +0900234 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800235}