// Copyright (c) 2012 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "content/renderer/text_input_client_observer.h" #include "base/memory/scoped_ptr.h" #include "content/common/text_input_client_messages.h" #include "content/renderer/pepper/pepper_plugin_instance_impl.h" #include "content/renderer/render_view_impl.h" #include "third_party/WebKit/public/platform/WebPoint.h" #include "third_party/WebKit/public/platform/WebRect.h" #include "third_party/WebKit/public/platform/WebString.h" #include "third_party/WebKit/public/web/WebLocalFrame.h" #include "third_party/WebKit/public/web/WebView.h" #include "third_party/WebKit/public/web/mac/WebSubstringUtil.h" #include "ui/gfx/rect.h" namespace content { TextInputClientObserver::TextInputClientObserver(RenderViewImpl* render_view) : RenderViewObserver(render_view), render_view_impl_(render_view) { } TextInputClientObserver::~TextInputClientObserver() { } bool TextInputClientObserver::OnMessageReceived(const IPC::Message& message) { bool handled = true; IPC_BEGIN_MESSAGE_MAP(TextInputClientObserver, message) IPC_MESSAGE_HANDLER(TextInputClientMsg_StringAtPoint, OnStringAtPoint) IPC_MESSAGE_HANDLER(TextInputClientMsg_CharacterIndexForPoint, OnCharacterIndexForPoint) IPC_MESSAGE_HANDLER(TextInputClientMsg_FirstRectForCharacterRange, OnFirstRectForCharacterRange) IPC_MESSAGE_HANDLER(TextInputClientMsg_StringForRange, OnStringForRange) IPC_MESSAGE_UNHANDLED(handled = false) IPC_END_MESSAGE_MAP() return handled; } blink::WebView* TextInputClientObserver::webview() { return render_view()->GetWebView(); } void TextInputClientObserver::OnStringAtPoint(gfx::Point point) { #if defined(OS_MACOSX) blink::WebPoint baselinePoint; NSAttributedString* string = blink::WebSubstringUtil::attributedWordAtPoint( webview(), point, baselinePoint); scoped_ptr<const mac::AttributedStringCoder::EncodedString> encoded( mac::AttributedStringCoder::Encode(string)); Send(new TextInputClientReplyMsg_GotStringAtPoint( routing_id(), *encoded.get(), baselinePoint)); #else NOTIMPLEMENTED(); #endif } void TextInputClientObserver::OnCharacterIndexForPoint(gfx::Point point) { blink::WebPoint web_point(point); size_t index = webview()->focusedFrame()->characterIndexForPoint(web_point); Send(new TextInputClientReplyMsg_GotCharacterIndexForPoint(routing_id(), index)); } void TextInputClientObserver::OnFirstRectForCharacterRange(gfx::Range range) { gfx::Rect rect; #if defined(ENABLE_PLUGINS) if (render_view_impl_->focused_pepper_plugin()) { rect = render_view_impl_->focused_pepper_plugin()->GetCaretBounds(); } else #endif { blink::WebFrame* frame = webview()->focusedFrame(); if (frame) { blink::WebRect web_rect; frame->firstRectForCharacterRange(range.start(), range.length(), web_rect); rect = web_rect; } } Send(new TextInputClientReplyMsg_GotFirstRectForRange(routing_id(), rect)); } void TextInputClientObserver::OnStringForRange(gfx::Range range) { #if defined(OS_MACOSX) NSAttributedString* string = nil; blink::WebLocalFrame* frame = webview()->focusedFrame()->toWebLocalFrame(); if (frame) { string = blink::WebSubstringUtil::attributedSubstringInRange( frame, range.start(), range.length()); } scoped_ptr<const mac::AttributedStringCoder::EncodedString> encoded( mac::AttributedStringCoder::Encode(string)); Send(new TextInputClientReplyMsg_GotStringForRange(routing_id(), *encoded.get())); #else NOTIMPLEMENTED(); #endif } } // namespace content