Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(448)

Unified Diff: remoting/host/single_window_input_injector_mac.mm

Issue 422503004: Adding ability to stream windows and inject events to them (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@master
Patch Set: Added comments, removed extraneous commented out code, and reformatted. Created 6 years, 4 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
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

Powered by Google App Engine
This is Rietveld 408576698