// 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 "remoting/host/clipboard.h" #include #include "base/bind.h" #include "base/logging.h" #include "base/message_loop.h" #include "remoting/host/linux/x_server_clipboard.h" #include "remoting/proto/event.pb.h" #include "remoting/protocol/clipboard_stub.h" namespace remoting { // This code is expected to be called on the desktop thread only. class ClipboardLinux : public Clipboard, public MessageLoopForIO::Watcher { public: ClipboardLinux(); virtual ~ClipboardLinux(); // Clipboard interface. virtual void Start( scoped_ptr client_clipboard) OVERRIDE; virtual void InjectClipboardEvent( const protocol::ClipboardEvent& event) OVERRIDE; virtual void Stop() OVERRIDE; // MessageLoopForIO::Watcher interface. virtual void OnFileCanReadWithoutBlocking(int fd) OVERRIDE; virtual void OnFileCanWriteWithoutBlocking(int fd) OVERRIDE; private: void OnClipboardChanged(const std::string& mime_type, const std::string& data); void PumpXEvents(); scoped_ptr client_clipboard_; // Underlying X11 clipboard implementation. XServerClipboard x_server_clipboard_; // Connection to the X server, used by |x_server_clipboard_|. This is created // and owned by this class. Display* display_; // Watcher used to handle X11 events from |display_|. MessageLoopForIO::FileDescriptorWatcher x_connection_watcher_; DISALLOW_COPY_AND_ASSIGN(ClipboardLinux); }; ClipboardLinux::ClipboardLinux() : display_(NULL) { } ClipboardLinux::~ClipboardLinux() { Stop(); } void ClipboardLinux::Start( scoped_ptr client_clipboard) { // TODO(lambroslambrou): Share the X connection with EventExecutor. display_ = XOpenDisplay(NULL); if (!display_) { LOG(ERROR) << "Couldn't open X display"; return; } client_clipboard_.swap(client_clipboard); x_server_clipboard_.Init(display_, base::Bind(&ClipboardLinux::OnClipboardChanged, base::Unretained(this))); MessageLoopForIO::current()->WatchFileDescriptor( ConnectionNumber(display_), true, MessageLoopForIO::WATCH_READ, &x_connection_watcher_, this); PumpXEvents(); } void ClipboardLinux::InjectClipboardEvent( const protocol::ClipboardEvent& event) { x_server_clipboard_.SetClipboard(event.mime_type(), event.data()); } void ClipboardLinux::Stop() { client_clipboard_.reset(); x_connection_watcher_.StopWatchingFileDescriptor(); if (display_) { XCloseDisplay(display_); display_ = NULL; } } void ClipboardLinux::OnFileCanReadWithoutBlocking(int fd) { PumpXEvents(); } void ClipboardLinux::OnFileCanWriteWithoutBlocking(int fd) { } void ClipboardLinux::OnClipboardChanged(const std::string& mime_type, const std::string& data) { protocol::ClipboardEvent event; event.set_mime_type(mime_type); event.set_data(data); if (client_clipboard_.get()) { client_clipboard_->InjectClipboardEvent(event); } } void ClipboardLinux::PumpXEvents() { DCHECK(display_); while (XPending(display_)) { XEvent event; XNextEvent(display_, &event); x_server_clipboard_.ProcessXEvent(&event); } } scoped_ptr Clipboard::Create() { return scoped_ptr(new ClipboardLinux()); } } // namespace remoting