如何使用 Python 和 GTK 创建具有自定义文本和透明背景的状态图标/系统托盘图标?

Posted

技术标签:

【中文标题】如何使用 Python 和 GTK 创建具有自定义文本和透明背景的状态图标/系统托盘图标?【英文标题】:How do I create a Status Icon / System Tray Icon with custom text and transparent background using Python and GTK? 【发布时间】:2010-06-12 16:19:35 【问题描述】:

这是我迄今为止定义图标的代码:

icon_bg = gtk.gdk.pixbuf_new_from_file('gmail.png')
w, h = icon_bg.get_width(), icon_bg.get_height()
cmap = gtk.gdk.Colormap(gtk.gdk.visual_get_system(), False)

drawable = gtk.gdk.Pixmap(None, w, h, 24)
drawable.set_colormap = cmap
gc = drawable.new_gc()
drawable.draw_pixbuf(gc, icon_bg, 0, 0, 0, 0, w, h)

drawn_icon = gtk.gdk.Pixbuf(gtk.gdk.COLORSPACE_RGB, False, 8, w, h)
drawn_icon.get_from_drawable(drawable, cmap, 0, 0, 0, 0, w, h)
icon = gtk.status_icon_new_from_pixbuf(drawn_icon)

这可以将 png 放入图标中,但在两个方面存在不足。首先,透明度不起作用。如果我使用具有透明背景且图像居中的 22x22 png,我最终会在我的内部显示其他活动图标的部分,如下所示:

http://i237.photobucket.com/albums/ff311/Raugturi/22x22_image_with_transparency.png

它选择窃取的图标有点随机。有时它是 Dropbox 图标的一部分,有时是 NetworkManager Applet。

如果我改为使用此代码:

icon_bg = gtk.gdk.pixbuf_new_from_file('gmail.png')
w, h = icon_bg.get_width(), icon_bg.get_height()
cmap = gtk.gdk.Colormap(gtk.gdk.visual_get_system(), False)

drawable = gtk.gdk.Pixmap(None, w, h, 24)
drawable.set_colormap = cmap
gc = drawable.new_gc()
drawable.draw_pixbuf(gc, icon_bg, 0, 0, 0, 0, w, h)

drawn_icon = gtk.gdk.Pixbuf(gtk.gdk.COLORSPACE_RGB, False, 8, 22, 22)
drawn_icon.get_from_drawable(drawable, cmap, 0, 0, 3, 6, w, h)
icon = gtk.status_icon_new_from_pixbuf(drawn_icon)

只有 16x11 的图像去除了透明边缘,我最终得到的是:

http://i237.photobucket.com/albums/ff311/Raugturi/16x11_image_positioned_in_middle.png

那么我如何最终得到一个透明块,比如第一个不从其他图标中提取内容的块?

至于第二个问题,我需要先在图像上写字,然后再将其转换为图标。我尝试使用 draw_glyphs,它告诉我应该使用 Pango 布局/上下文。不幸的是,我能找到的所有 Pango 教程都处理实际的窗口,而不是状态图标。是否有适用于这个问题的 Pango 的好教程(也可能至少有一些关于如何告诉它使用什么字体的解释,因为我发现所有这些字体似乎都缺少这个,它不会写没有它的任何东西)。

注意:很抱歉缺少实际图片且只有一个有效链接,由于我缺乏声誉,显然这是一项防止垃圾邮件的功能。

【问题讨论】:

【参考方案1】:

我编写了这个工具(用 C 语言,但它被设计为一个单独的进程)可以与任何语言(perl、python、tcl、lua、c、c++、bash、ksh、...)一起使用,只需启动它带有工具提示和左右单击命令的任意数量的图标(仅受托盘大小的限制)。然后根据需要更改您的图标。处理图标最简单的方法是通过 SVG,因为它真的很容易包含其他图像(包括 png)并使用预格式化命令快速定义文本甚至复杂(或简单)形状。

#include <sys/inotify.h>
#include <gtk/gtk.h>
#define EVENT_SIZE  ( sizeof (struct inotify_event) )
#define EVENT_BUF_LEN     ( 1024 * ( EVENT_SIZE + 16 ) )

void leftclick(GtkStatusIcon *si, gpointer s) 
    popen(s,"r"); /* exec s */


void rightclick(GtkStatusIcon *si, guint b,guint a_t, gpointer s) 
    popen(s,"r");


void refresh(gpointer si, gint fd, GdkInputCondition c) 
    char buffer[EVENT_BUF_LEN];
    read( fd, buffer, EVENT_BUF_LEN ); /* we are just clearing it & do not care what event type */
    gtk_status_icon_set_from_file(si,gtk_status_icon_get_title(si)); /* redraws */


int main(int argc, char *argv[]) 
    GtkStatusIcon *si; int i=1, watch, fd;
    gtk_init (&argc, &argv); /* loop through icon, tooltip, click messages */
    while (i<argc)     fd = inotify_init(); /* get file descriptor to write on if image changes */
        si = gtk_status_icon_new_from_file(argv[i]); /* get a status icon widget */
        gtk_status_icon_set_title(si,argv[i]); /* hack to store the image path */
        watch = inotify_add_watch( fd, argv[i++], IN_CREATE | IN_MODIFY | IN_MOVED_FROM );
        gdk_input_add( fd, GDK_INPUT_READ, refresh, si ); /* inotify fd is ready for reading, refresh */
        gtk_status_icon_set_tooltip_text(si,argv[i++]);
        g_signal_connect(G_OBJECT(si), "activate", G_CALLBACK(leftclick), (gpointer) argv[i++]);
        g_signal_connect(G_OBJECT(si), "popup-menu", G_CALLBACK(rightclick), (gpointer) argv[i++]);
    
    gtk_main();

【讨论】:

【参考方案2】:

你有什么理由不能只使用gtk.StatusIcon.set_from_file?它适用于透明图像。

至于您的第二个问题,我非常怀疑是否可以在 gtk.StatusIcon 上使用 pango。 为什么要在图标上写文字?

【讨论】:

我正在编写一个应用程序来检查 Google 服务并将未读消息的计数放在状态图标上,这样您就可以一眼看出有多少消息。据我所知,这只给了我两个选择:在图标上写字或加载数字图像。如果可能的话,我宁愿写让用户自定义字体。 我想我想出了一个解决方案,但要到今晚才能对其进行测试。我应该能够加载背景图像,在其上写入,将其保存到临时文件中,然后在其上使用 set_from_file。 Raugturi,你找到解决办法了吗?现在正在尝试同样的事情。

以上是关于如何使用 Python 和 GTK 创建具有自定义文本和透明背景的状态图标/系统托盘图标?的主要内容,如果未能解决你的问题,请参考以下文章

如何开始使用自定义 GUI 小部件

GTK+ 3.0:如何将 Gtk.TreeStore 与自定义模型项一起使用?

如何使用 Python 中用 Vala 编写的自定义 GTK 小部件?

是否可以指示 `Gtk::TreeView` 显示自定义类型?

如何在 GTK 应用程序中使用自定义字体

如何在 GTK3 和 Python 2.7 中的图像对象上创建“点击事件”