• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright 2013 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 "chrome/browser/ui/libgtk2ui/app_indicator_icon.h"
6 
7 #include <gtk/gtk.h>
8 #include <dlfcn.h>
9 
10 #include "base/bind.h"
11 #include "base/file_util.h"
12 #include "base/memory/ref_counted_memory.h"
13 #include "base/strings/stringprintf.h"
14 #include "base/strings/utf_string_conversions.h"
15 #include "base/threading/sequenced_worker_pool.h"
16 #include "chrome/browser/ui/libgtk2ui/menu_util.h"
17 #include "content/public/browser/browser_thread.h"
18 #include "ui/base/models/menu_model.h"
19 #include "ui/gfx/image/image_skia.h"
20 
21 namespace {
22 
23 typedef enum {
24   APP_INDICATOR_CATEGORY_APPLICATION_STATUS,
25   APP_INDICATOR_CATEGORY_COMMUNICATIONS,
26   APP_INDICATOR_CATEGORY_SYSTEM_SERVICES,
27   APP_INDICATOR_CATEGORY_HARDWARE,
28   APP_INDICATOR_CATEGORY_OTHER
29 } AppIndicatorCategory;
30 
31 typedef enum {
32   APP_INDICATOR_STATUS_PASSIVE,
33   APP_INDICATOR_STATUS_ACTIVE,
34   APP_INDICATOR_STATUS_ATTENTION
35 } AppIndicatorStatus;
36 
37 typedef AppIndicator* (*app_indicator_new_func)(const gchar* id,
38                                                 const gchar* icon_name,
39                                                 AppIndicatorCategory category);
40 
41 typedef AppIndicator* (*app_indicator_new_with_path_func)(
42     const gchar* id,
43     const gchar* icon_name,
44     AppIndicatorCategory category,
45     const gchar* icon_theme_path);
46 
47 typedef void (*app_indicator_set_status_func)(AppIndicator* self,
48                                               AppIndicatorStatus status);
49 
50 typedef void (*app_indicator_set_attention_icon_full_func)(
51     AppIndicator* self,
52     const gchar* icon_name,
53     const gchar* icon_desc);
54 
55 typedef void (*app_indicator_set_menu_func)(AppIndicator* self, GtkMenu* menu);
56 
57 typedef void (*app_indicator_set_icon_full_func)(AppIndicator* self,
58                                                  const gchar* icon_name,
59                                                  const gchar* icon_desc);
60 
61 typedef void (*app_indicator_set_icon_theme_path_func)(
62     AppIndicator* self,
63     const gchar* icon_theme_path);
64 
65 bool g_attempted_load = false;
66 bool g_opened = false;
67 
68 // Retrieved functions from libappindicator.
69 app_indicator_new_func app_indicator_new = NULL;
70 app_indicator_new_with_path_func app_indicator_new_with_path = NULL;
71 app_indicator_set_status_func app_indicator_set_status = NULL;
72 app_indicator_set_attention_icon_full_func
73     app_indicator_set_attention_icon_full = NULL;
74 app_indicator_set_menu_func app_indicator_set_menu = NULL;
75 app_indicator_set_icon_full_func app_indicator_set_icon_full = NULL;
76 app_indicator_set_icon_theme_path_func app_indicator_set_icon_theme_path = NULL;
77 
EnsureMethodsLoaded()78 void EnsureMethodsLoaded() {
79   if (g_attempted_load)
80     return;
81 
82   g_attempted_load = true;
83 
84   void* indicator_lib = dlopen("libappindicator.so", RTLD_LAZY);
85   if (!indicator_lib) {
86     indicator_lib = dlopen("libappindicator.so.1", RTLD_LAZY);
87   }
88   if (!indicator_lib) {
89     indicator_lib = dlopen("libappindicator.so.0", RTLD_LAZY);
90   }
91   if (!indicator_lib) {
92     return;
93   }
94 
95   g_opened = true;
96 
97   app_indicator_new = reinterpret_cast<app_indicator_new_func>(
98       dlsym(indicator_lib, "app_indicator_new"));
99 
100   app_indicator_new_with_path =
101       reinterpret_cast<app_indicator_new_with_path_func>(
102           dlsym(indicator_lib, "app_indicator_new_with_path"));
103 
104   app_indicator_set_status = reinterpret_cast<app_indicator_set_status_func>(
105       dlsym(indicator_lib, "app_indicator_set_status"));
106 
107   app_indicator_set_attention_icon_full =
108       reinterpret_cast<app_indicator_set_attention_icon_full_func>(
109           dlsym(indicator_lib, "app_indicator_set_attention_icon_full"));
110 
111   app_indicator_set_menu = reinterpret_cast<app_indicator_set_menu_func>(
112       dlsym(indicator_lib, "app_indicator_set_menu"));
113 
114   app_indicator_set_icon_full =
115       reinterpret_cast<app_indicator_set_icon_full_func>(
116           dlsym(indicator_lib, "app_indicator_set_icon_full"));
117 
118   app_indicator_set_icon_theme_path =
119       reinterpret_cast<app_indicator_set_icon_theme_path_func>(
120           dlsym(indicator_lib, "app_indicator_set_icon_theme_path"));
121 }
122 
CreateTempImageFile(gfx::ImageSkia * image_ptr,int icon_change_count,std::string id)123 base::FilePath CreateTempImageFile(gfx::ImageSkia* image_ptr,
124                                    int icon_change_count,
125                                    std::string id) {
126   scoped_ptr<gfx::ImageSkia> image(image_ptr);
127 
128   scoped_refptr<base::RefCountedMemory> png_data =
129       gfx::Image(*image.get()).As1xPNGBytes();
130   if (png_data->size() == 0) {
131     // If the bitmap could not be encoded to PNG format, skip it.
132     LOG(WARNING) << "Could not encode icon";
133     return base::FilePath();
134   }
135 
136   base::FilePath temp_dir;
137   base::FilePath new_file_path;
138 
139   // Create a new temporary directory for each image since using a single
140   // temporary directory seems to have issues when changing icons in quick
141   // succession.
142   if (!base::CreateNewTempDirectory(base::FilePath::StringType(), &temp_dir))
143     return base::FilePath();
144   new_file_path =
145       temp_dir.Append(id + base::StringPrintf("_%d.png", icon_change_count));
146   int bytes_written =
147       file_util::WriteFile(new_file_path,
148                            reinterpret_cast<const char*>(png_data->front()),
149                            png_data->size());
150 
151   if (bytes_written != static_cast<int>(png_data->size()))
152     return base::FilePath();
153   return new_file_path;
154 }
155 
DeleteTempImagePath(const base::FilePath & icon_file_path)156 void DeleteTempImagePath(const base::FilePath& icon_file_path) {
157   if (icon_file_path.empty())
158     return;
159   base::DeleteFile(icon_file_path, true);
160 }
161 
162 }  // namespace
163 
164 namespace libgtk2ui {
165 
AppIndicatorIcon(std::string id,const gfx::ImageSkia & image,const base::string16 & tool_tip)166 AppIndicatorIcon::AppIndicatorIcon(std::string id,
167                                    const gfx::ImageSkia& image,
168                                    const base::string16& tool_tip)
169     : id_(id),
170       icon_(NULL),
171       gtk_menu_(NULL),
172       menu_model_(NULL),
173       icon_change_count_(0),
174       block_activation_(false),
175       weak_factory_(this) {
176   EnsureMethodsLoaded();
177   tool_tip_ = UTF16ToUTF8(tool_tip);
178   SetImage(image);
179 }
~AppIndicatorIcon()180 AppIndicatorIcon::~AppIndicatorIcon() {
181   if (icon_) {
182     app_indicator_set_status(icon_, APP_INDICATOR_STATUS_PASSIVE);
183     if (gtk_menu_)
184       DestroyMenu();
185     g_object_unref(icon_);
186     content::BrowserThread::GetBlockingPool()->PostTask(
187         FROM_HERE,
188         base::Bind(&DeleteTempImagePath, icon_file_path_.DirName()));
189   }
190 }
191 
192 // static
CouldOpen()193 bool AppIndicatorIcon::CouldOpen() {
194   EnsureMethodsLoaded();
195   return g_opened;
196 }
197 
SetImage(const gfx::ImageSkia & image)198 void AppIndicatorIcon::SetImage(const gfx::ImageSkia& image) {
199   if (!g_opened)
200     return;
201 
202   ++icon_change_count_;
203 
204   // We create a deep copy of the image since it may have been freed by the time
205   // it's accessed in the other thread.
206   scoped_ptr<gfx::ImageSkia> safe_image(image.DeepCopy());
207   base::PostTaskAndReplyWithResult(
208       content::BrowserThread::GetBlockingPool()
209           ->GetTaskRunnerWithShutdownBehavior(
210                 base::SequencedWorkerPool::SKIP_ON_SHUTDOWN).get(),
211       FROM_HERE,
212       base::Bind(&CreateTempImageFile,
213                  safe_image.release(),
214                  icon_change_count_,
215                  id_),
216       base::Bind(&AppIndicatorIcon::SetImageFromFile,
217                  weak_factory_.GetWeakPtr()));
218 }
219 
SetPressedImage(const gfx::ImageSkia & image)220 void AppIndicatorIcon::SetPressedImage(const gfx::ImageSkia& image) {
221   // Ignore pressed images, since the standard on Linux is to not highlight
222   // pressed status icons.
223 }
224 
SetToolTip(const base::string16 & tool_tip)225 void AppIndicatorIcon::SetToolTip(const base::string16& tool_tip) {
226   DCHECK(!tool_tip_.empty());
227   tool_tip_ = UTF16ToUTF8(tool_tip);
228 
229   // We can set the click action label only if the icon exists. Also we only
230   // need to update the label if it is shown and it's only shown if we are sure
231   // that there is a click action or if there is no menu.
232   if (icon_ && (delegate()->HasClickAction() || menu_model_ == NULL)) {
233     GList* children = gtk_container_get_children(GTK_CONTAINER(gtk_menu_));
234     for (GList* child = children; child; child = g_list_next(child))
235       if (g_object_get_data(G_OBJECT(child->data), "click-action-item") !=
236           NULL) {
237         gtk_menu_item_set_label(GTK_MENU_ITEM(child->data),
238                                 tool_tip_.c_str());
239         break;
240       }
241     g_list_free(children);
242   }
243 }
244 
UpdatePlatformContextMenu(ui::MenuModel * model)245 void AppIndicatorIcon::UpdatePlatformContextMenu(ui::MenuModel* model) {
246   if (!g_opened)
247     return;
248 
249   if (gtk_menu_) {
250     DestroyMenu();
251   }
252   menu_model_ = model;
253 
254   // The icon is created asynchronously so it might not exist when the menu is
255   // set.
256   if (icon_)
257     SetMenu();
258 }
259 
RefreshPlatformContextMenu()260 void AppIndicatorIcon::RefreshPlatformContextMenu() {
261   gtk_container_foreach(
262       GTK_CONTAINER(gtk_menu_), SetMenuItemInfo, &block_activation_);
263 }
264 
SetImageFromFile(const base::FilePath & icon_file_path)265 void AppIndicatorIcon::SetImageFromFile(const base::FilePath& icon_file_path) {
266   DCHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::UI));
267   if (icon_file_path.empty())
268     return;
269 
270   base::FilePath old_path = icon_file_path_;
271   icon_file_path_ = icon_file_path;
272 
273   std::string icon_name =
274       icon_file_path_.BaseName().RemoveExtension().value();
275   std::string icon_dir = icon_file_path_.DirName().value();
276   if (!icon_) {
277     icon_ =
278         app_indicator_new_with_path(id_.c_str(),
279                                     icon_name.c_str(),
280                                     APP_INDICATOR_CATEGORY_APPLICATION_STATUS,
281                                     icon_dir.c_str());
282     app_indicator_set_status(icon_, APP_INDICATOR_STATUS_ACTIVE);
283     SetMenu();
284   } else {
285     // Currently we are creating a new temp directory every time the icon is
286     // set. So we need to set the directory each time.
287     app_indicator_set_icon_theme_path(icon_, icon_dir.c_str());
288     app_indicator_set_icon_full(icon_, icon_name.c_str(), "icon");
289 
290     // Delete previous icon directory.
291     content::BrowserThread::GetBlockingPool()->PostTask(
292         FROM_HERE,
293         base::Bind(&DeleteTempImagePath, old_path.DirName()));
294   }
295 }
296 
SetMenu()297 void AppIndicatorIcon::SetMenu() {
298   gtk_menu_ = gtk_menu_new();
299 
300   if (delegate()->HasClickAction() || menu_model_ == NULL) {
301     CreateClickActionReplacement();
302     if (menu_model_) {
303       // Add separator before the other menu items.
304       GtkWidget* menu_item = gtk_separator_menu_item_new();
305       gtk_widget_show(menu_item);
306       gtk_menu_shell_append(GTK_MENU_SHELL(gtk_menu_), menu_item);
307     }
308   }
309   if (menu_model_) {
310     BuildSubmenuFromModel(menu_model_,
311                           gtk_menu_,
312                           G_CALLBACK(OnMenuItemActivatedThunk),
313                           &block_activation_,
314                           this);
315     RefreshPlatformContextMenu();
316   }
317   app_indicator_set_menu(icon_, GTK_MENU(gtk_menu_));
318 }
319 
CreateClickActionReplacement()320 void AppIndicatorIcon::CreateClickActionReplacement() {
321   DCHECK(!tool_tip_.empty());
322 
323   // Add "click replacement menu item".
324   GtkWidget* menu_item = gtk_menu_item_new_with_mnemonic(tool_tip_.c_str());
325   g_object_set_data(
326       G_OBJECT(menu_item), "click-action-item", GINT_TO_POINTER(1));
327   g_signal_connect(menu_item, "activate", G_CALLBACK(OnClickThunk), this);
328   gtk_widget_show(menu_item);
329   gtk_menu_shell_prepend(GTK_MENU_SHELL(gtk_menu_), menu_item);
330 }
331 
DestroyMenu()332 void AppIndicatorIcon::DestroyMenu() {
333   gtk_widget_destroy(gtk_menu_);
334   gtk_menu_ = NULL;
335   menu_model_ = NULL;
336 }
337 
OnClick(GtkWidget * menu_item)338 void AppIndicatorIcon::OnClick(GtkWidget* menu_item) {
339   if (delegate())
340     delegate()->OnClick();
341 }
342 
OnMenuItemActivated(GtkWidget * menu_item)343 void AppIndicatorIcon::OnMenuItemActivated(GtkWidget* menu_item) {
344   if (block_activation_)
345     return;
346 
347   ui::MenuModel* model = ModelForMenuItem(GTK_MENU_ITEM(menu_item));
348   if (!model) {
349     // There won't be a model for "native" submenus like the "Input Methods"
350     // context menu. We don't need to handle activation messages for submenus
351     // anyway, so we can just return here.
352     DCHECK(gtk_menu_item_get_submenu(GTK_MENU_ITEM(menu_item)));
353     return;
354   }
355 
356   // The activate signal is sent to radio items as they get deselected;
357   // ignore it in this case.
358   if (GTK_IS_RADIO_MENU_ITEM(menu_item) &&
359       !gtk_check_menu_item_get_active(GTK_CHECK_MENU_ITEM(menu_item))) {
360     return;
361   }
362 
363   int id;
364   if (!GetMenuItemID(menu_item, &id))
365     return;
366 
367   // The menu item can still be activated by hotkeys even if it is disabled.
368   if (menu_model_->IsEnabledAt(id))
369     ExecuteCommand(model, id);
370 }
371 
372 }  // namespace libgtk2ui
373