Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #include "content/browser/renderer_host/ime_adapter_android.h" | 5 #include "content/browser/renderer_host/ime_adapter_android.h" |
| 6 | 6 |
| 7 #include <android/input.h> | 7 #include <android/input.h> |
| 8 | 8 |
| 9 #include "base/android/jni_android.h" | 9 #include "base/android/jni_android.h" |
| 10 #include "base/android/jni_string.h" | 10 #include "base/android/jni_string.h" |
| 11 #include "base/android/scoped_java_ref.h" | 11 #include "base/android/scoped_java_ref.h" |
| 12 #include "base/strings/utf_string_conversions.h" | 12 #include "base/strings/utf_string_conversions.h" |
| 13 #include "base/time/time.h" | 13 #include "base/time/time.h" |
| 14 #include "content/browser/frame_host/frame_tree.h" | |
| 15 #include "content/browser/frame_host/frame_tree_node.h" | |
| 16 #include "content/browser/frame_host/render_frame_host_impl.h" | |
| 17 #include "content/browser/renderer_host/render_view_host_impl.h" | |
| 14 #include "content/browser/renderer_host/render_widget_host_impl.h" | 18 #include "content/browser/renderer_host/render_widget_host_impl.h" |
| 15 #include "content/browser/renderer_host/render_widget_host_view_android.h" | 19 #include "content/browser/renderer_host/render_widget_host_view_android.h" |
| 16 #include "content/common/view_messages.h" | 20 #include "content/common/view_messages.h" |
| 21 #include "content/public/browser/browser_thread.h" | |
| 17 #include "content/public/browser/native_web_keyboard_event.h" | 22 #include "content/public/browser/native_web_keyboard_event.h" |
| 18 #include "jni/ImeAdapter_jni.h" | 23 #include "jni/ImeAdapter_jni.h" |
| 19 #include "third_party/WebKit/public/web/WebCompositionUnderline.h" | 24 #include "third_party/WebKit/public/web/WebCompositionUnderline.h" |
| 20 #include "third_party/WebKit/public/web/WebInputEvent.h" | 25 #include "third_party/WebKit/public/web/WebInputEvent.h" |
| 21 | 26 |
| 22 using base::android::AttachCurrentThread; | 27 using base::android::AttachCurrentThread; |
| 23 using base::android::ConvertJavaStringToUTF16; | 28 using base::android::ConvertJavaStringToUTF16; |
| 24 | 29 |
| 25 namespace content { | 30 namespace content { |
| 26 namespace { | 31 namespace { |
| (...skipping 65 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 92 | 97 |
| 93 bool ImeAdapterAndroid::SendSyntheticKeyEvent(JNIEnv*, | 98 bool ImeAdapterAndroid::SendSyntheticKeyEvent(JNIEnv*, |
| 94 jobject, | 99 jobject, |
| 95 int type, | 100 int type, |
| 96 long time_ms, | 101 long time_ms, |
| 97 int key_code, | 102 int key_code, |
| 98 int text) { | 103 int text) { |
| 99 NativeWebKeyboardEvent event(static_cast<blink::WebInputEvent::Type>(type), | 104 NativeWebKeyboardEvent event(static_cast<blink::WebInputEvent::Type>(type), |
| 100 0 /* modifiers */, time_ms / 1000.0, key_code, | 105 0 /* modifiers */, time_ms / 1000.0, key_code, |
| 101 text, false /* is_system_key */); | 106 text, false /* is_system_key */); |
| 102 rwhva_->SendKeyEvent(event); | 107 BrowserThread::PostTask( |
| 108 BrowserThread::UI, FROM_HERE, | |
| 109 base::Bind(&ImeAdapterAndroid::SendSyntheticKeyEventOnUI, this, event)); | |
|
aurimas (slooooooooow)
2014/03/07 18:04:40
I was under impression that most of these calls in
jam
2014/03/07 18:15:45
sure that works too. i didn't know how they're cal
| |
| 103 return true; | 110 return true; |
| 104 } | 111 } |
| 105 | 112 |
| 106 bool ImeAdapterAndroid::SendKeyEvent(JNIEnv* env, jobject, | 113 bool ImeAdapterAndroid::SendKeyEvent(JNIEnv* env, jobject, |
| 107 jobject original_key_event, | 114 jobject original_key_event, |
| 108 int action, int modifiers, | 115 int action, int modifiers, |
| 109 long time_ms, int key_code, | 116 long time_ms, int key_code, |
| 110 bool is_system_key, int unicode_char) { | 117 bool is_system_key, int unicode_char) { |
| 111 NativeWebKeyboardEvent event = NativeWebKeyboardEventFromKeyEvent( | 118 NativeWebKeyboardEvent event = NativeWebKeyboardEventFromKeyEvent( |
| 112 env, original_key_event, action, modifiers, | 119 env, original_key_event, action, modifiers, |
| 113 time_ms, key_code, is_system_key, unicode_char); | 120 time_ms, key_code, is_system_key, unicode_char); |
| 114 bool key_down_text_insertion = | 121 bool key_down_text_insertion = |
| 115 event.type == blink::WebInputEvent::RawKeyDown && event.text[0]; | 122 event.type == blink::WebInputEvent::RawKeyDown && event.text[0]; |
| 116 // If we are going to follow up with a synthetic Char event, then that's the | 123 // If we are going to follow up with a synthetic Char event, then that's the |
| 117 // one we expect to test if it's handled or unhandled, so skip handling the | 124 // one we expect to test if it's handled or unhandled, so skip handling the |
| 118 // "real" event in the browser. | 125 // "real" event in the browser. |
| 119 event.skip_in_browser = key_down_text_insertion; | 126 event.skip_in_browser = key_down_text_insertion; |
| 120 rwhva_->SendKeyEvent(event); | 127 |
| 121 if (key_down_text_insertion) { | 128 // Send a Char event, but without an os_event since we don't want to |
| 122 // Send a Char event, but without an os_event since we don't want to | 129 // roundtrip back to java such synthetic event. |
| 123 // roundtrip back to java such synthetic event. | 130 NativeWebKeyboardEvent char_event(blink::WebInputEvent::Char, modifiers, |
| 124 NativeWebKeyboardEvent char_event(blink::WebInputEvent::Char, modifiers, | 131 time_ms, key_code, unicode_char, |
| 125 time_ms, key_code, unicode_char, | 132 is_system_key); |
| 126 is_system_key); | 133 char_event.skip_in_browser = key_down_text_insertion; |
| 127 char_event.skip_in_browser = key_down_text_insertion; | 134 |
| 128 rwhva_->SendKeyEvent(char_event); | 135 BrowserThread::PostTask( |
| 129 } | 136 BrowserThread::UI, FROM_HERE, |
| 137 base::Bind(&ImeAdapterAndroid::SendKeyEventOnUI, this, event, | |
| 138 char_event, key_down_text_insertion)); | |
| 139 | |
| 130 return true; | 140 return true; |
| 131 } | 141 } |
| 132 | 142 |
| 133 void ImeAdapterAndroid::SetComposingText(JNIEnv* env, jobject, jstring text, | 143 void ImeAdapterAndroid::SetComposingText(JNIEnv* env, jobject, jstring text, |
| 134 int new_cursor_pos) { | 144 int new_cursor_pos) { |
| 135 RenderWidgetHostImpl* rwhi = GetRenderWidgetHostImpl(); | |
| 136 if (!rwhi) | |
| 137 return; | |
| 138 | |
| 139 base::string16 text16 = ConvertJavaStringToUTF16(env, text); | 145 base::string16 text16 = ConvertJavaStringToUTF16(env, text); |
| 140 std::vector<blink::WebCompositionUnderline> underlines; | 146 BrowserThread::PostTask( |
| 141 underlines.push_back( | 147 BrowserThread::UI, FROM_HERE, |
| 142 blink::WebCompositionUnderline(0, text16.length(), SK_ColorBLACK, | 148 base::Bind(&ImeAdapterAndroid::SetComposingTextOnUI, this, text16, |
| 143 false)); | 149 new_cursor_pos)); |
| 144 // new_cursor_position is as described in the Android API for | |
| 145 // InputConnection#setComposingText, whereas the parameters for | |
| 146 // ImeSetComposition are relative to the start of the composition. | |
| 147 if (new_cursor_pos > 0) | |
| 148 new_cursor_pos = text16.length() + new_cursor_pos - 1; | |
| 149 | |
| 150 rwhi->ImeSetComposition(text16, underlines, new_cursor_pos, new_cursor_pos); | |
| 151 } | 150 } |
| 152 | 151 |
| 153 void ImeAdapterAndroid::CommitText(JNIEnv* env, jobject, jstring text) { | 152 void ImeAdapterAndroid::CommitText(JNIEnv* env, jobject, jstring text) { |
| 154 RenderWidgetHostImpl* rwhi = GetRenderWidgetHostImpl(); | |
| 155 if (!rwhi) | |
| 156 return; | |
| 157 | |
| 158 base::string16 text16 = ConvertJavaStringToUTF16(env, text); | 153 base::string16 text16 = ConvertJavaStringToUTF16(env, text); |
| 159 rwhi->ImeConfirmComposition(text16, gfx::Range::InvalidRange(), false); | 154 BrowserThread::PostTask( |
| 155 BrowserThread::UI, FROM_HERE, | |
| 156 base::Bind(&ImeAdapterAndroid::CommitTextOnUI, this, text16)); | |
| 160 } | 157 } |
| 161 | 158 |
| 162 void ImeAdapterAndroid::FinishComposingText(JNIEnv* env, jobject) { | 159 void ImeAdapterAndroid::FinishComposingText(JNIEnv* env, jobject) { |
| 163 RenderWidgetHostImpl* rwhi = GetRenderWidgetHostImpl(); | 160 BrowserThread::PostTask( |
| 164 if (!rwhi) | 161 BrowserThread::UI, FROM_HERE, |
| 165 return; | 162 base::Bind(&ImeAdapterAndroid::FinishComposingTextsOnUI, this)); |
| 166 | |
| 167 rwhi->ImeConfirmComposition(base::string16(), gfx::Range::InvalidRange(), | |
| 168 true); | |
| 169 } | 163 } |
| 170 | 164 |
| 171 void ImeAdapterAndroid::AttachImeAdapter(JNIEnv* env, jobject java_object) { | 165 void ImeAdapterAndroid::AttachImeAdapter(JNIEnv* env, jobject java_object) { |
| 172 java_ime_adapter_ = JavaObjectWeakGlobalRef(env, java_object); | 166 java_ime_adapter_ = JavaObjectWeakGlobalRef(env, java_object); |
| 173 } | 167 } |
| 174 | 168 |
| 175 void ImeAdapterAndroid::CancelComposition() { | 169 void ImeAdapterAndroid::CancelComposition() { |
| 176 base::android::ScopedJavaLocalRef<jobject> obj = | 170 base::android::ScopedJavaLocalRef<jobject> obj = |
| 177 java_ime_adapter_.get(AttachCurrentThread()); | 171 java_ime_adapter_.get(AttachCurrentThread()); |
| 178 if (!obj.is_null()) | 172 if (!obj.is_null()) |
| 179 Java_ImeAdapter_cancelComposition(AttachCurrentThread(), obj.obj()); | 173 Java_ImeAdapter_cancelComposition(AttachCurrentThread(), obj.obj()); |
| 180 } | 174 } |
| 181 | 175 |
| 182 void ImeAdapterAndroid::FocusedNodeChanged(bool is_editable_node) { | 176 void ImeAdapterAndroid::FocusedNodeChanged(bool is_editable_node) { |
| 183 base::android::ScopedJavaLocalRef<jobject> obj = | 177 base::android::ScopedJavaLocalRef<jobject> obj = |
| 184 java_ime_adapter_.get(AttachCurrentThread()); | 178 java_ime_adapter_.get(AttachCurrentThread()); |
| 185 if (!obj.is_null()) { | 179 if (!obj.is_null()) { |
| 186 Java_ImeAdapter_focusedNodeChanged(AttachCurrentThread(), | 180 Java_ImeAdapter_focusedNodeChanged(AttachCurrentThread(), |
| 187 obj.obj(), | 181 obj.obj(), |
| 188 is_editable_node); | 182 is_editable_node); |
| 189 } | 183 } |
| 190 } | 184 } |
| 191 | 185 |
| 186 void ImeAdapterAndroid::RenderWidgetGone() { | |
| 187 rwhva_ = NULL; | |
| 188 } | |
| 189 | |
| 192 void ImeAdapterAndroid::SetEditableSelectionOffsets(JNIEnv*, jobject, | 190 void ImeAdapterAndroid::SetEditableSelectionOffsets(JNIEnv*, jobject, |
| 193 int start, int end) { | 191 int start, int end) { |
| 192 BrowserThread::PostTask( | |
| 193 BrowserThread::UI, FROM_HERE, | |
| 194 base::Bind(&ImeAdapterAndroid::SetEditableSelectionOffsetsOnUI, this, | |
| 195 start, end)); | |
| 196 } | |
| 197 | |
| 198 void ImeAdapterAndroid::SetComposingRegion(JNIEnv*, jobject, | |
| 199 int start, int end) { | |
| 200 BrowserThread::PostTask( | |
| 201 BrowserThread::UI, FROM_HERE, | |
| 202 base::Bind(&ImeAdapterAndroid::SetComposingRegiontOnUI, this, | |
| 203 start, end)); | |
| 204 } | |
| 205 | |
| 206 void ImeAdapterAndroid::DeleteSurroundingText(JNIEnv*, jobject, | |
| 207 int before, int after) { | |
| 208 BrowserThread::PostTask( | |
| 209 BrowserThread::UI, FROM_HERE, | |
| 210 base::Bind(&ImeAdapterAndroid::DeleteSurroundingTextOnUI, this, | |
| 211 before, after)); | |
| 212 } | |
| 213 | |
| 214 void ImeAdapterAndroid::Unselect(JNIEnv* env, jobject) { | |
| 215 BrowserThread::PostTask( | |
| 216 BrowserThread::UI, FROM_HERE, | |
| 217 base::Bind(&ImeAdapterAndroid::UnselectOnUI, this)); | |
| 218 } | |
| 219 | |
| 220 void ImeAdapterAndroid::SelectAll(JNIEnv* env, jobject) { | |
| 221 BrowserThread::PostTask( | |
| 222 BrowserThread::UI, FROM_HERE, | |
| 223 base::Bind(&ImeAdapterAndroid::SelectAllOnUI, this)); | |
| 224 } | |
| 225 | |
| 226 void ImeAdapterAndroid::Cut(JNIEnv* env, jobject) { | |
| 227 BrowserThread::PostTask( | |
| 228 BrowserThread::UI, FROM_HERE, | |
| 229 base::Bind(&ImeAdapterAndroid::CutOnUI, this)); | |
| 230 } | |
| 231 | |
| 232 void ImeAdapterAndroid::Copy(JNIEnv* env, jobject) { | |
| 233 BrowserThread::PostTask( | |
| 234 BrowserThread::UI, FROM_HERE, | |
| 235 base::Bind(&ImeAdapterAndroid::CopyOnUI, this)); | |
| 236 } | |
| 237 | |
| 238 void ImeAdapterAndroid::Paste(JNIEnv* env, jobject) { | |
| 239 BrowserThread::PostTask( | |
| 240 BrowserThread::UI, FROM_HERE, | |
| 241 base::Bind(&ImeAdapterAndroid::PasteOnUI, this)); | |
| 242 } | |
| 243 | |
| 244 void ImeAdapterAndroid::ResetImeAdapter(JNIEnv* env, jobject) { | |
| 245 java_ime_adapter_.reset(); | |
| 246 } | |
| 247 | |
| 248 void ImeAdapterAndroid::SendSyntheticKeyEventOnUI( | |
| 249 const NativeWebKeyboardEvent& event) { | |
| 250 rwhva_->SendKeyEvent(event); | |
| 251 } | |
| 252 | |
| 253 void ImeAdapterAndroid::SendKeyEventOnUI( | |
| 254 const NativeWebKeyboardEvent& event, | |
| 255 const NativeWebKeyboardEvent& char_event, | |
| 256 bool key_down_text_insertion) { | |
| 257 rwhva_->SendKeyEvent(event); | |
| 258 if (key_down_text_insertion) | |
| 259 rwhva_->SendKeyEvent(char_event); | |
| 260 } | |
| 261 | |
| 262 void ImeAdapterAndroid::SetComposingTextOnUI(const base::string16& text16, | |
| 263 int new_cursor_pos) { | |
| 264 RenderWidgetHostImpl* rwhi = GetRenderWidgetHostImpl(); | |
| 265 if (!rwhi) | |
| 266 return; | |
| 267 | |
| 268 std::vector<blink::WebCompositionUnderline> underlines; | |
| 269 underlines.push_back( | |
| 270 blink::WebCompositionUnderline(0, text16.length(), SK_ColorBLACK, | |
| 271 false)); | |
| 272 // new_cursor_position is as described in the Android API for | |
| 273 // InputConnection#setComposingText, whereas the parameters for | |
| 274 // ImeSetComposition are relative to the start of the composition. | |
| 275 if (new_cursor_pos > 0) | |
| 276 new_cursor_pos = text16.length() + new_cursor_pos - 1; | |
| 277 | |
| 278 rwhi->ImeSetComposition(text16, underlines, new_cursor_pos, new_cursor_pos); | |
| 279 } | |
| 280 | |
| 281 void ImeAdapterAndroid::CommitTextOnUI(const base::string16& text16) { | |
| 282 RenderWidgetHostImpl* rwhi = GetRenderWidgetHostImpl(); | |
| 283 if (!rwhi) | |
| 284 return; | |
| 285 | |
| 286 rwhi->ImeConfirmComposition(text16, gfx::Range::InvalidRange(), false); | |
| 287 } | |
| 288 | |
| 289 void ImeAdapterAndroid::FinishComposingTextsOnUI() { | |
| 290 RenderWidgetHostImpl* rwhi = GetRenderWidgetHostImpl(); | |
| 291 if (!rwhi) | |
| 292 return; | |
| 293 | |
| 294 rwhi->ImeConfirmComposition(base::string16(), gfx::Range::InvalidRange(), | |
| 295 true); | |
| 296 } | |
| 297 | |
| 298 void ImeAdapterAndroid::SetEditableSelectionOffsetsOnUI(int start, int end) { | |
| 194 RenderWidgetHostImpl* rwhi = GetRenderWidgetHostImpl(); | 299 RenderWidgetHostImpl* rwhi = GetRenderWidgetHostImpl(); |
| 195 if (!rwhi) | 300 if (!rwhi) |
| 196 return; | 301 return; |
| 197 | 302 |
| 198 rwhi->Send(new ViewMsg_SetEditableSelectionOffsets(rwhi->GetRoutingID(), | 303 rwhi->Send(new ViewMsg_SetEditableSelectionOffsets(rwhi->GetRoutingID(), |
| 199 start, end)); | 304 start, end)); |
| 200 } | 305 } |
| 201 | 306 |
| 202 void ImeAdapterAndroid::SetComposingRegion(JNIEnv*, jobject, | 307 void ImeAdapterAndroid::SetComposingRegiontOnUI(int start, int end) { |
| 203 int start, int end) { | |
| 204 RenderWidgetHostImpl* rwhi = GetRenderWidgetHostImpl(); | 308 RenderWidgetHostImpl* rwhi = GetRenderWidgetHostImpl(); |
| 205 if (!rwhi) | 309 if (!rwhi) |
| 206 return; | 310 return; |
| 207 | 311 |
| 208 std::vector<blink::WebCompositionUnderline> underlines; | 312 std::vector<blink::WebCompositionUnderline> underlines; |
| 209 underlines.push_back( | 313 underlines.push_back( |
| 210 blink::WebCompositionUnderline(0, end - start, SK_ColorBLACK, false)); | 314 blink::WebCompositionUnderline(0, end - start, SK_ColorBLACK, false)); |
| 211 | 315 |
| 212 rwhi->Send(new ViewMsg_SetCompositionFromExistingText( | 316 rwhi->Send(new ViewMsg_SetCompositionFromExistingText( |
| 213 rwhi->GetRoutingID(), start, end, underlines)); | 317 rwhi->GetRoutingID(), start, end, underlines)); |
| 214 } | 318 } |
| 215 | 319 |
| 216 void ImeAdapterAndroid::DeleteSurroundingText(JNIEnv*, jobject, | 320 void ImeAdapterAndroid::DeleteSurroundingTextOnUI(int before, int after) { |
| 217 int before, int after) { | |
| 218 RenderWidgetHostImpl* rwhi = GetRenderWidgetHostImpl(); | 321 RenderWidgetHostImpl* rwhi = GetRenderWidgetHostImpl(); |
| 219 if (!rwhi) | 322 if (!rwhi) |
| 220 return; | 323 return; |
| 221 | 324 |
| 222 rwhi->Send(new ViewMsg_ExtendSelectionAndDelete(rwhi->GetRoutingID(), | 325 rwhi->Send(new ViewMsg_ExtendSelectionAndDelete(rwhi->GetRoutingID(), |
| 223 before, after)); | 326 before, after)); |
| 224 } | 327 } |
| 225 | 328 |
| 226 void ImeAdapterAndroid::Unselect(JNIEnv* env, jobject) { | 329 void ImeAdapterAndroid::UnselectOnUI() { |
| 227 RenderWidgetHostImpl* rwhi = GetRenderWidgetHostImpl(); | 330 RenderWidgetHostImpl* rwhi = GetRenderWidgetHostImpl(); |
| 228 if (!rwhi) | 331 if (rwhi) |
| 229 return; | 332 rwhi->Unselect(); |
| 230 | |
| 231 rwhi->Unselect(); | |
| 232 } | 333 } |
| 233 | 334 |
| 234 void ImeAdapterAndroid::SelectAll(JNIEnv* env, jobject) { | 335 void ImeAdapterAndroid::SelectAllOnUI() { |
| 235 RenderWidgetHostImpl* rwhi = GetRenderWidgetHostImpl(); | 336 RenderWidgetHostImpl* rwhi = GetRenderWidgetHostImpl(); |
| 236 if (!rwhi) | 337 if (rwhi) |
| 237 return; | 338 rwhi->SelectAll(); |
| 238 | |
| 239 rwhi->SelectAll(); | |
| 240 } | 339 } |
| 241 | 340 |
| 242 void ImeAdapterAndroid::Cut(JNIEnv* env, jobject) { | 341 void ImeAdapterAndroid::CutOnUI() { |
| 243 RenderWidgetHostImpl* rwhi = GetRenderWidgetHostImpl(); | 342 RenderFrameHost* rfh = GetFocusedFrame(); |
| 244 if (!rwhi) | 343 if (rfh) |
| 245 return; | 344 rfh->Cut(); |
| 246 | |
| 247 rwhi->Cut(); | |
| 248 } | 345 } |
| 249 | 346 |
| 250 void ImeAdapterAndroid::Copy(JNIEnv* env, jobject) { | 347 void ImeAdapterAndroid::CopyOnUI() { |
| 251 RenderWidgetHostImpl* rwhi = GetRenderWidgetHostImpl(); | 348 RenderFrameHost* rfh = GetFocusedFrame(); |
| 252 if (!rwhi) | 349 if (rfh) |
| 253 return; | 350 rfh->Copy(); |
| 254 | |
| 255 rwhi->Copy(); | |
| 256 } | 351 } |
| 257 | 352 |
| 258 void ImeAdapterAndroid::Paste(JNIEnv* env, jobject) { | 353 void ImeAdapterAndroid::PasteOnUI() { |
| 259 RenderWidgetHostImpl* rwhi = GetRenderWidgetHostImpl(); | 354 RenderFrameHost* rfh = GetFocusedFrame(); |
| 260 if (!rwhi) | 355 if (rfh) |
| 261 return; | 356 rfh->Paste(); |
| 262 | |
| 263 rwhi->Paste(); | |
| 264 } | |
| 265 | |
| 266 void ImeAdapterAndroid::ResetImeAdapter(JNIEnv* env, jobject) { | |
| 267 java_ime_adapter_.reset(); | |
| 268 } | 357 } |
| 269 | 358 |
| 270 RenderWidgetHostImpl* ImeAdapterAndroid::GetRenderWidgetHostImpl() { | 359 RenderWidgetHostImpl* ImeAdapterAndroid::GetRenderWidgetHostImpl() { |
| 271 DCHECK(rwhva_); | 360 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); |
| 361 if (!rwhva_) | |
| 362 return NULL; | |
| 363 | |
| 272 RenderWidgetHost* rwh = rwhva_->GetRenderWidgetHost(); | 364 RenderWidgetHost* rwh = rwhva_->GetRenderWidgetHost(); |
| 273 if (!rwh) | 365 if (!rwh) |
| 274 return NULL; | 366 return NULL; |
| 275 | 367 |
| 276 return RenderWidgetHostImpl::From(rwh); | 368 return RenderWidgetHostImpl::From(rwh); |
| 277 } | 369 } |
| 278 | 370 |
| 371 RenderFrameHost* ImeAdapterAndroid::GetFocusedFrame() { | |
| 372 RenderWidgetHostImpl* rwh = GetRenderWidgetHostImpl(); | |
| 373 if (!rwh) | |
| 374 return NULL; | |
| 375 if (!rwh->IsRenderView()) | |
| 376 return NULL; | |
| 377 RenderViewHost* rvh = RenderViewHost::From(rwh); | |
| 378 RenderFrameHostImpl* rfh = | |
| 379 static_cast<RenderFrameHostImpl*>(rvh->GetMainFrame()); | |
| 380 FrameTreeNode* focused_frame = | |
| 381 rfh->frame_tree_node()->frame_tree()->GetFocusedFrame(); | |
| 382 if (!focused_frame) | |
| 383 return NULL; | |
| 384 | |
| 385 return focused_frame->current_frame_host(); | |
| 386 } | |
| 387 | |
| 279 } // namespace content | 388 } // namespace content |
| OLD | NEW |