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 3 // found in the LICENSE file. 4 5 #include <gtk/gtk.h> 6 #include <math.h> 7 8 #include "base/compiler_specific.h" 9 #include "base/logging.h" 10 #include "base/strings/string_util.h" 11 #include "base/strings/utf_string_conversions.h" 12 #include "remoting/base/string_resources.h" 13 #include "remoting/host/client_session_control.h" 14 #include "remoting/host/host_window.h" 15 #include "ui/base/gtk/gtk_signal.h" 16 #include "ui/base/l10n/l10n_util.h" 17 18 namespace remoting { 19 20 namespace { 21 22 class DisconnectWindowGtk : public HostWindow { 23 public: 24 DisconnectWindowGtk(); 25 virtual ~DisconnectWindowGtk(); 26 27 // HostWindow overrides. 28 virtual void Start( 29 const base::WeakPtr<ClientSessionControl>& client_session_control) 30 OVERRIDE; 31 32 private: 33 CHROMEGTK_CALLBACK_1(DisconnectWindowGtk, gboolean, OnDelete, GdkEvent*); 34 CHROMEGTK_CALLBACK_0(DisconnectWindowGtk, void, OnClicked); 35 CHROMEGTK_CALLBACK_1(DisconnectWindowGtk, gboolean, OnConfigure, 36 GdkEventConfigure*); 37 CHROMEGTK_CALLBACK_1(DisconnectWindowGtk, gboolean, OnButtonPress, 38 GdkEventButton*); 39 40 // Used to disconnect the client session. 41 base::WeakPtr<ClientSessionControl> client_session_control_; 42 43 GtkWidget* disconnect_window_; 44 GtkWidget* message_; 45 GtkWidget* button_; 46 47 // Used to distinguish resize events from other types of "configure-event" 48 // notifications. 49 int current_width_; 50 int current_height_; 51 52 DISALLOW_COPY_AND_ASSIGN(DisconnectWindowGtk); 53 }; 54 55 // Helper function for creating a rectangular path with rounded corners, as 56 // Cairo doesn't have this facility. |radius| is the arc-radius of each 57 // corner. The bounding rectangle extends from (0, 0) to (width, height). 58 void AddRoundRectPath(cairo_t* cairo_context, int width, int height, 59 int radius) { 60 cairo_new_sub_path(cairo_context); 61 cairo_arc(cairo_context, width - radius, radius, radius, -M_PI_2, 0); 62 cairo_arc(cairo_context, width - radius, height - radius, radius, 0, M_PI_2); 63 cairo_arc(cairo_context, radius, height - radius, radius, M_PI_2, 2 * M_PI_2); 64 cairo_arc(cairo_context, radius, radius, radius, 2 * M_PI_2, 3 * M_PI_2); 65 cairo_close_path(cairo_context); 66 } 67 68 DisconnectWindowGtk::DisconnectWindowGtk() 69 : disconnect_window_(NULL), 70 current_width_(0), 71 current_height_(0) { 72 } 73 74 DisconnectWindowGtk::~DisconnectWindowGtk() { 75 DCHECK(CalledOnValidThread()); 76 77 if (disconnect_window_) { 78 gtk_widget_destroy(disconnect_window_); 79 disconnect_window_ = NULL; 80 } 81 } 82 83 void DisconnectWindowGtk::Start( 84 const base::WeakPtr<ClientSessionControl>& client_session_control) { 85 DCHECK(CalledOnValidThread()); 86 DCHECK(!client_session_control_.get()); 87 DCHECK(client_session_control.get()); 88 DCHECK(!disconnect_window_); 89 90 client_session_control_ = client_session_control; 91 92 // Create the window. 93 disconnect_window_ = gtk_window_new(GTK_WINDOW_TOPLEVEL); 94 GtkWindow* window = GTK_WINDOW(disconnect_window_); 95 96 g_signal_connect(disconnect_window_, "delete-event", 97 G_CALLBACK(OnDeleteThunk), this); 98 gtk_window_set_title(window, 99 l10n_util::GetStringUTF8(IDR_PRODUCT_NAME).c_str()); 100 gtk_window_set_resizable(window, FALSE); 101 102 // Try to keep the window always visible. 103 gtk_window_stick(window); 104 gtk_window_set_keep_above(window, TRUE); 105 106 // Remove window titlebar. 107 gtk_window_set_decorated(window, FALSE); 108 109 // In case the titlebar is still there, try to remove some of the buttons. 110 // Utility windows have no minimize button or taskbar presence. 111 gtk_window_set_type_hint(window, GDK_WINDOW_TYPE_HINT_UTILITY); 112 gtk_window_set_deletable(window, FALSE); 113 114 // Allow custom rendering of the background pixmap. 115 gtk_widget_set_app_paintable(disconnect_window_, TRUE); 116 117 // Handle window resizing, to regenerate the background pixmap and window 118 // shape bitmap. The stored width & height need to be initialized here 119 // in case the window is created a second time (the size of the previous 120 // window would be remembered, preventing the generation of bitmaps for the 121 // new window). 122 current_height_ = current_width_ = 0; 123 g_signal_connect(disconnect_window_, "configure-event", 124 G_CALLBACK(OnConfigureThunk), this); 125 126 // Handle mouse events to allow the user to drag the window around. 127 gtk_widget_set_events(disconnect_window_, GDK_BUTTON_PRESS_MASK); 128 g_signal_connect(disconnect_window_, "button-press-event", 129 G_CALLBACK(OnButtonPressThunk), this); 130 131 // All magic numbers taken from screen shots provided by UX. 132 // The alignment sets narrow margins at the top and bottom, compared with 133 // left and right. The left margin is made larger to accommodate the 134 // window movement gripper. 135 GtkWidget* align = gtk_alignment_new(0, 0, 1, 1); 136 gtk_alignment_set_padding(GTK_ALIGNMENT(align), 8, 8, 24, 12); 137 gtk_container_add(GTK_CONTAINER(window), align); 138 139 GtkWidget* button_row = gtk_hbox_new(FALSE, 12); 140 gtk_container_add(GTK_CONTAINER(align), button_row); 141 142 button_ = gtk_button_new_with_label( 143 l10n_util::GetStringUTF8(IDR_STOP_SHARING_BUTTON).c_str()); 144 gtk_box_pack_end(GTK_BOX(button_row), button_, FALSE, FALSE, 0); 145 146 g_signal_connect(button_, "clicked", G_CALLBACK(OnClickedThunk), this); 147 148 message_ = gtk_label_new(NULL); 149 gtk_box_pack_end(GTK_BOX(button_row), message_, FALSE, FALSE, 0); 150 151 // Override any theme setting for the text color, so that the text is 152 // readable against the window's background pixmap. 153 PangoAttrList* attributes = pango_attr_list_new(); 154 PangoAttribute* text_color = pango_attr_foreground_new(0, 0, 0); 155 pango_attr_list_insert(attributes, text_color); 156 gtk_label_set_attributes(GTK_LABEL(message_), attributes); 157 pango_attr_list_unref(attributes); 158 159 gtk_widget_show_all(disconnect_window_); 160 161 // Extract the user name from the JID. 162 std::string client_jid = client_session_control_->client_jid(); 163 base::string16 username = 164 UTF8ToUTF16(client_jid.substr(0, client_jid.find('/'))); 165 gtk_label_set_text( 166 GTK_LABEL(message_), 167 l10n_util::GetStringFUTF8(IDR_MESSAGE_SHARED, username).c_str()); 168 gtk_window_present(window); 169 } 170 171 void DisconnectWindowGtk::OnClicked(GtkWidget* button) { 172 DCHECK(CalledOnValidThread()); 173 174 if (client_session_control_.get()) 175 client_session_control_->DisconnectSession(); 176 } 177 178 gboolean DisconnectWindowGtk::OnDelete(GtkWidget* window, 179 GdkEvent* event) { 180 DCHECK(CalledOnValidThread()); 181 182 if (client_session_control_.get()) 183 client_session_control_->DisconnectSession(); 184 return TRUE; 185 } 186 187 gboolean DisconnectWindowGtk::OnConfigure(GtkWidget* widget, 188 GdkEventConfigure* event) { 189 DCHECK(CalledOnValidThread()); 190 191 // Only generate bitmaps if the size has actually changed. 192 if (event->width == current_width_ && event->height == current_height_) 193 return FALSE; 194 195 current_width_ = event->width; 196 current_height_ = event->height; 197 198 // Create the depth 1 pixmap for the window shape. 199 GdkPixmap* shape_mask = gdk_pixmap_new(NULL, current_width_, current_height_, 200 1); 201 cairo_t* cairo_context = gdk_cairo_create(shape_mask); 202 203 // Set the arc radius for the corners. 204 const int kCornerRadius = 6; 205 206 // Initialize the whole bitmap to be transparent. 207 cairo_set_source_rgba(cairo_context, 0, 0, 0, 0); 208 cairo_set_operator(cairo_context, CAIRO_OPERATOR_SOURCE); 209 cairo_paint(cairo_context); 210 211 // Paint an opaque round rect covering the whole area (leaving the extreme 212 // corners transparent). 213 cairo_set_source_rgba(cairo_context, 1, 1, 1, 1); 214 cairo_set_operator(cairo_context, CAIRO_OPERATOR_SOURCE); 215 AddRoundRectPath(cairo_context, current_width_, current_height_, 216 kCornerRadius); 217 cairo_fill(cairo_context); 218 219 cairo_destroy(cairo_context); 220 gdk_window_shape_combine_mask(widget->window, shape_mask, 0, 0); 221 g_object_unref(shape_mask); 222 223 // Create a full-color pixmap for the window background image. 224 GdkPixmap* background = gdk_pixmap_new(NULL, current_width_, current_height_, 225 24); 226 cairo_context = gdk_cairo_create(background); 227 228 // Paint the whole bitmap one color. 229 cairo_set_source_rgb(cairo_context, 0.91, 0.91, 0.91); 230 cairo_paint(cairo_context); 231 232 // Paint the round-rectangle edge. 233 cairo_set_source_rgb(cairo_context, 0.13, 0.69, 0.11); 234 cairo_set_line_width(cairo_context, 6); 235 AddRoundRectPath(cairo_context, current_width_, current_height_, 236 kCornerRadius); 237 cairo_stroke(cairo_context); 238 239 // Render the window-gripper. In order for a straight line to light up 240 // single pixels, Cairo requires the coordinates to have fractional 241 // components of 0.5 (so the "/ 2" is a deliberate integer division). 242 double gripper_top = current_height_ / 2 - 10.5; 243 double gripper_bottom = current_height_ / 2 + 10.5; 244 cairo_set_line_width(cairo_context, 1); 245 246 double x = 12.5; 247 cairo_set_source_rgb(cairo_context, 0.70, 0.70, 0.70); 248 cairo_move_to(cairo_context, x, gripper_top); 249 cairo_line_to(cairo_context, x, gripper_bottom); 250 cairo_stroke(cairo_context); 251 x += 3; 252 cairo_move_to(cairo_context, x, gripper_top); 253 cairo_line_to(cairo_context, x, gripper_bottom); 254 cairo_stroke(cairo_context); 255 256 x -= 2; 257 cairo_set_source_rgb(cairo_context, 0.97, 0.97, 0.97); 258 cairo_move_to(cairo_context, x, gripper_top); 259 cairo_line_to(cairo_context, x, gripper_bottom); 260 cairo_stroke(cairo_context); 261 x += 3; 262 cairo_move_to(cairo_context, x, gripper_top); 263 cairo_line_to(cairo_context, x, gripper_bottom); 264 cairo_stroke(cairo_context); 265 266 cairo_destroy(cairo_context); 267 268 gdk_window_set_back_pixmap(widget->window, background, FALSE); 269 g_object_unref(background); 270 gdk_window_invalidate_rect(widget->window, NULL, TRUE); 271 272 return FALSE; 273 } 274 275 gboolean DisconnectWindowGtk::OnButtonPress(GtkWidget* widget, 276 GdkEventButton* event) { 277 DCHECK(CalledOnValidThread()); 278 279 gtk_window_begin_move_drag(GTK_WINDOW(disconnect_window_), 280 event->button, 281 event->x_root, 282 event->y_root, 283 event->time); 284 return FALSE; 285 } 286 287 } // namespace 288 289 // static 290 scoped_ptr<HostWindow> HostWindow::CreateDisconnectWindow() { 291 return scoped_ptr<HostWindow>(new DisconnectWindowGtk()); 292 } 293 294 } // namespace remoting 295