Chromium Code Reviews| Index: remoting/host/single_window_input_injector_mac.mm |
| diff --git a/remoting/host/single_window_input_injector_mac.mm b/remoting/host/single_window_input_injector_mac.mm |
| new file mode 100644 |
| index 0000000000000000000000000000000000000000..cf33ab42c6e38781583e0613e89eba3ba9c2490b |
| --- /dev/null |
| +++ b/remoting/host/single_window_input_injector_mac.mm |
| @@ -0,0 +1,167 @@ |
| +// Copyright 2014 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 "remoting/host/single_window_input_injector.h" |
| + |
| +#include <ApplicationServices/ApplicationServices.h> |
| +#include <Carbon/Carbon.h> |
| + |
| +#include "base/mac/scoped_cftyperef.h" |
| +#include "remoting/proto/internal.pb.h" |
|
Wez
2014/08/06 04:10:16
I think you want to include the event.pb.h header,
ronakvora do not use
2014/08/06 20:56:19
Done.
|
| +#include "third_party/webrtc/modules/desktop_capture/mac/desktop_configuration.h" |
| + |
| +namespace remoting { |
| + |
| +using protocol::ClipboardEvent; |
| +using protocol::KeyEvent; |
| +using protocol::TextEvent; |
| +using protocol::MouseEvent; |
| + |
| +// A class to generate events to a window on Mac. |
|
Wez
2014/08/06 04:10:16
nit: Suggest:
"Implements InputInjector to direct
ronakvora do not use
2014/08/06 20:56:19
Done.
|
| +class SingleWindowInputInjectorMac : public SingleWindowInputInjector { |
| + public: |
| + SingleWindowInputInjectorMac( |
| + webrtc::WindowId window_id, |
| + scoped_ptr<InputInjector> input_injector); |
| + virtual ~SingleWindowInputInjectorMac(); |
| + |
| + // InputInjector interface. |
| + virtual void Start( |
| + scoped_ptr<protocol::ClipboardStub> client_clipboard) OVERRIDE; |
| + virtual void InjectKeyEvent(const KeyEvent& event) OVERRIDE; |
| + virtual void InjectTextEvent(const TextEvent& event) OVERRIDE; |
| + virtual void InjectMouseEvent(const MouseEvent& event) OVERRIDE; |
| + virtual void InjectClipboardEvent(const ClipboardEvent& event) OVERRIDE; |
| + |
| + private: |
| + CGRect FindCGRectOfWindow(); |
| + |
| + CGWindowID window_id_; |
| + scoped_ptr<InputInjector> input_injector_; |
| + |
| + DISALLOW_COPY_AND_ASSIGN(SingleWindowInputInjectorMac); |
| +}; |
| + |
| +SingleWindowInputInjectorMac::SingleWindowInputInjectorMac( |
| + webrtc::WindowId window_id, |
| + scoped_ptr<InputInjector> input_injector): |
| + window_id_((CGWindowID)window_id), |
| + input_injector_(input_injector.Pass()) { |
| +} |
| + |
| +SingleWindowInputInjectorMac::~SingleWindowInputInjectorMac() { |
| +} |
| + |
| +void SingleWindowInputInjectorMac::Start( |
| + scoped_ptr<protocol::ClipboardStub> client_clipboard) { |
| + input_injector_->Start(client_clipboard.Pass()); |
| +} |
| + |
| +void SingleWindowInputInjectorMac::InjectKeyEvent(const KeyEvent& event) { |
| + input_injector_->InjectKeyEvent(event); |
| +} |
| + |
| +void SingleWindowInputInjectorMac::InjectTextEvent(const TextEvent& event) { |
| + input_injector_->InjectTextEvent(event); |
| +} |
| + |
| +void SingleWindowInputInjectorMac::InjectMouseEvent(const MouseEvent& event) { |
| + if (event.has_x() && event.has_y()) { |
| + CGRect windowRect = FindCGRectOfWindow(); |
| + if (CGRectIsNull(windowRect)) { |
| + LOG(ERROR) << "Failing silently and just forwarding event"; |
|
Wez
2014/08/06 04:10:16
nit: Suggest "Window rect is null, so ignoring Mou
ronakvora do not use
2014/08/06 20:56:20
Done.
|
| + input_injector_->InjectMouseEvent(event); |
| + return; |
| + } |
|
Wez
2014/08/06 04:10:16
nit: Blank line after this, to separate it from th
ronakvora do not use
2014/08/06 20:56:19
Done.
|
| + webrtc::MacDesktopConfiguration desktop_config = |
| + webrtc::MacDesktopConfiguration::GetCurrent( |
| + webrtc::MacDesktopConfiguration::TopLeftOrigin); |
| + |
| + // Create a vector that has the origin of the window. |
| + webrtc::DesktopVector mouse_pos(CGRectGetMinX(windowRect), |
|
Wez
2014/08/06 04:10:16
mouse_pos -> window_origin, then?
ronakvora do not use
2014/08/06 20:56:19
Yep, you're right. We were doing it this way in ou
|
| + CGRectGetMinY(windowRect)); |
| + |
| + // Subtract coordinates that are already added in for multiple |
| + // desktops. They get added in already in InputInjector. |
|
Wez
2014/08/06 04:10:16
Suggest: "The underlying InputInjector expects coo
ronakvora do not use
2014/08/06 20:56:19
Done.
|
| + mouse_pos.subtract( |
| + webrtc::DesktopVector(desktop_config.pixel_bounds.left(), |
| + desktop_config.pixel_bounds.top())); |
| + |
| + // Create a new event with coordinates that are in respect to the window. |
| + // We must make sure we are taking into account the fact that when we |
| + // find the window on the host it returns is coordinates in Density |
|
Lambros
2014/08/05 22:47:25
s/is/its
s/conver/convert
ronakvora do not use
2014/08/06 20:56:19
Done.
|
| + // Independent coordinates. We have to conver to Density Dependent |
| + // because InputInjector assumes Density Dependent coordinates in the |
| + // MouseEvent. |
| + MouseEvent modified_event(event); |
| + modified_event.set_x( |
| + event.x() + mouse_pos.x() * desktop_config.dip_to_pixel_scale); |
| + modified_event.set_y( |
| + event.y() + mouse_pos.y() * desktop_config.dip_to_pixel_scale); |
| + input_injector_->InjectMouseEvent(modified_event); |
| + } else { |
| + input_injector_->InjectMouseEvent(event); |
| + } |
| +} |
| + |
| +void SingleWindowInputInjectorMac::InjectClipboardEvent( |
| + const ClipboardEvent& event) { |
| + input_injector_->InjectClipboardEvent(event); |
| +} |
| + |
| +// This method is used if we are capturing a window instead of a screen. |
| +// It finds the rectangle of the window we are streaming using |
| +// |window_id_|. The InputInjector can then use this rectangle |
| +// to translate the input event to coordinates of the window rather |
| +// than the screen. |
| +CGRect SingleWindowInputInjectorMac::FindCGRectOfWindow() { |
| + CGRect rect; |
| + CGWindowID ids[1] = {window_id_}; |
| + base::ScopedCFTypeRef<CFArrayRef> window_id_array( |
| + CFArrayCreate(NULL, reinterpret_cast<const void **>(&ids), 1, NULL)); |
| + |
| + base::ScopedCFTypeRef<CFArrayRef> window_array( |
| + CGWindowListCreateDescriptionFromArray(window_id_array)); |
| + |
| + if (window_array == NULL || CFArrayGetCount(window_array) == 0) { |
| + // Could not find the window. It might have been closed. |
| + LOG(ERROR) << "Specified window to stream not found for id: " |
| + << window_id_; |
| + return CGRectNull; |
| + } |
| + |
| + // We don't use ScopedCFTypeRef for |window| because the |
|
Lambros
2014/08/05 22:47:25
I don't think this comment is needed.
ronakvora do not use
2014/08/06 20:56:19
When I had a discussion with Wez, he asked me to d
|
| + // CFDictionaryRef returned by CFArrayGetValueAtIndex is owned by |
| + // window_array. The same is true of the |bounds|. |
| + CFDictionaryRef window = |
| + reinterpret_cast<CFDictionaryRef>( |
| + CFArrayGetValueAtIndex(window_array, 0)); |
| + |
| + if (CFDictionaryContainsKey(window, kCGWindowBounds)) { |
| + CFDictionaryRef bounds = |
| + reinterpret_cast<CFDictionaryRef>( |
| + CFDictionaryGetValue(window, kCGWindowBounds)); |
| + if (bounds) { |
| + CGRectMakeWithDictionaryRepresentation(bounds, &rect); |
| + } else { |
| + return CGRectNull; |
| + } |
| + } else { |
| + return CGRectNull; |
| + } |
| + |
| + return rect; |
| +} |
| + |
| +scoped_ptr<InputInjector> SingleWindowInputInjector::Create( |
| + webrtc::WindowId window_id, |
| + scoped_ptr<InputInjector> input_injector) { |
| + scoped_ptr<SingleWindowInputInjectorMac>injector( |
| + new SingleWindowInputInjectorMac(window_id, input_injector.Pass())); |
| + return injector.PassAs<InputInjector>(); |
| +} |
| + |
| + |
| + |
| +} // namespace remoting |