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