1 /*
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 
17 package com.android.internal.widget;
18 
19 import android.os.Bundle;
20 import android.text.Editable;
21 import android.text.Spanned;
22 import android.text.method.KeyListener;
23 import android.text.style.SuggestionSpan;
24 import android.util.Log;
25 import android.view.inputmethod.BaseInputConnection;
26 import android.view.inputmethod.CompletionInfo;
27 import android.view.inputmethod.CorrectionInfo;
28 import android.view.inputmethod.ExtractedText;
29 import android.view.inputmethod.ExtractedTextRequest;
30 import android.view.inputmethod.InputConnection;
31 import android.widget.TextView;
32 
33 public 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 
39     // 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 
EditableInputConnection(TextView textview)44     public EditableInputConnection(TextView textview) {
45         super(textview, true);
46         mTextView = textview;
47     }
48 
49     @Override
getEditable()50     public Editable getEditable() {
51         TextView tv = mTextView;
52         if (tv != null) {
53             return tv.getEditableText();
54         }
55         return null;
56     }
57 
58     @Override
beginBatchEdit()59     public boolean beginBatchEdit() {
60         synchronized(this) {
61             if (mBatchEditNesting >= 0) {
62                 mTextView.beginBatchEdit();
63                 mBatchEditNesting++;
64                 return true;
65             }
66         }
67         return false;
68     }
69 
70     @Override
endBatchEdit()71     public boolean endBatchEdit() {
72         synchronized(this) {
73             if (mBatchEditNesting > 0) {
74                 // When the connection is reset by the InputMethodManager and reportFinish
75                 // 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;
84     }
85 
86     @Override
closeConnection()87     public void closeConnection() {
88         super.closeConnection();
89         synchronized(this) {
90             while (mBatchEditNesting > 0) {
91                 endBatchEdit();
92             }
93             // Will prevent any further calls to begin or endBatchEdit
94             mBatchEditNesting = -1;
95         }
96     }
97 
98     @Override
clearMetaKeyStates(int states)99     public boolean clearMetaKeyStates(int states) {
100         final Editable content = getEditable();
101         if (content == null) return false;
102         KeyListener kl = mTextView.getKeyListener();
103         if (kl != null) {
104             try {
105                 kl.clearMetaKeyState(mTextView, content, states);
106             } catch (AbstractMethodError e) {
107                 // This is an old listener that doesn't implement the
108                 // new method.
109             }
110         }
111         return true;
112     }
113 
114     @Override
commitCompletion(CompletionInfo text)115     public boolean commitCompletion(CompletionInfo text) {
116         if (DEBUG) Log.v(TAG, "commitCompletion " + text);
117         mTextView.beginBatchEdit();
118         mTextView.onCommitCompletion(text);
119         mTextView.endBatchEdit();
120         return true;
121     }
122 
123     /**
124      * Calls the {@link TextView#onCommitCorrection} method of the associated TextView.
125      */
126     @Override
commitCorrection(CorrectionInfo correctionInfo)127     public boolean commitCorrection(CorrectionInfo correctionInfo) {
128         if (DEBUG) Log.v(TAG, "commitCorrection" + correctionInfo);
129         mTextView.beginBatchEdit();
130         mTextView.onCommitCorrection(correctionInfo);
131         mTextView.endBatchEdit();
132         return true;
133     }
134 
135     @Override
performEditorAction(int actionCode)136     public boolean performEditorAction(int actionCode) {
137         if (DEBUG) Log.v(TAG, "performEditorAction " + actionCode);
138         mTextView.onEditorAction(actionCode);
139         return true;
140     }
141 
142     @Override
performContextMenuAction(int id)143     public boolean performContextMenuAction(int id) {
144         if (DEBUG) Log.v(TAG, "performContextMenuAction " + id);
145         mTextView.beginBatchEdit();
146         mTextView.onTextContextMenuItem(id);
147         mTextView.endBatchEdit();
148         return true;
149     }
150 
151     @Override
getExtractedText(ExtractedTextRequest request, int flags)152     public ExtractedText getExtractedText(ExtractedTextRequest request, int flags) {
153         if (mTextView != null) {
154             ExtractedText et = new ExtractedText();
155             if (mTextView.extractText(request, et)) {
156                 if ((flags&GET_EXTRACTED_TEXT_MONITOR) != 0) {
157                     mTextView.setExtracting(request);
158                 }
159                 return et;
160             }
161         }
162         return null;
163     }
164 
165     @Override
performPrivateCommand(String action, Bundle data)166     public boolean performPrivateCommand(String action, Bundle data) {
167         mTextView.onPrivateIMECommand(action, data);
168         return true;
169     }
170 
171     @Override
commitText(CharSequence text, int newCursorPosition)172     public boolean commitText(CharSequence text, int newCursorPosition) {
173         if (mTextView == null) {
174             return super.commitText(text, newCursorPosition);
175         }
176         if (text instanceof Spanned) {
177             Spanned spanned = ((Spanned) text);
178             SuggestionSpan[] spans = spanned.getSpans(0, text.length(), SuggestionSpan.class);
179             mIMM.registerSuggestionSpansForNotification(spans);
180         }
181 
182         mTextView.resetErrorChangedFlag();
183         boolean success = super.commitText(text, newCursorPosition);
184         mTextView.hideErrorIfUnchanged();
185 
186         return success;
187     }
188 
189     @Override
requestCursorUpdates(int cursorUpdateMode)190     public boolean requestCursorUpdates(int cursorUpdateMode) {
191         if (DEBUG) Log.v(TAG, "requestUpdateCursorAnchorInfo " + cursorUpdateMode);
192 
193         // It is possible that any other bit is used as a valid flag in a future release.
194         // We should reject the entire request in such a case.
195         final int KNOWN_FLAGS_MASK = InputConnection.CURSOR_UPDATE_IMMEDIATE |
196                 InputConnection.CURSOR_UPDATE_MONITOR;
197         final int unknownFlags = cursorUpdateMode & ~KNOWN_FLAGS_MASK;
198         if (unknownFlags != 0) {
199             if (DEBUG) {
200                 Log.d(TAG, "Rejecting requestUpdateCursorAnchorInfo due to unknown flags." +
201                         " cursorUpdateMode=" + cursorUpdateMode +
202                         " unknownFlags=" + unknownFlags);
203             }
204             return false;
205         }
206 
207         if (mIMM == null) {
208             // In this case, TYPE_CURSOR_ANCHOR_INFO is not handled.
209             // TODO: Return some notification code rather than false to indicate method that
210             // CursorAnchorInfo is temporarily unavailable.
211             return false;
212         }
213         mIMM.setUpdateCursorAnchorInfoMode(cursorUpdateMode);
214         if ((cursorUpdateMode & InputConnection.CURSOR_UPDATE_IMMEDIATE) != 0) {
215             if (mTextView == null) {
216                 // In this case, FLAG_CURSOR_ANCHOR_INFO_IMMEDIATE is silently ignored.
217                 // TODO: Return some notification code for the input method that indicates
218                 // FLAG_CURSOR_ANCHOR_INFO_IMMEDIATE is ignored.
219             } else if (mTextView.isInLayout()) {
220                 // In this case, the view hierarchy is currently undergoing a layout pass.
221                 // IMM#updateCursorAnchorInfo is supposed to be called soon after the layout
222                 // pass is finished.
223             } else {
224                 // This will schedule a layout pass of the view tree, and the layout event
225                 // eventually triggers IMM#updateCursorAnchorInfo.
226                 mTextView.requestLayout();
227             }
228         }
229         return true;
230     }
231 }
232