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

2 投票
2 回答
2562 浏览
提问于 2025-04-15 23:53

这是我目前用来定义图标的代码:

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的图标。

如果我改用这个代码:

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教程(最好还能解释一下如何指定字体,因为我找到的教程似乎都没有这部分内容,而没有字体的话,它什么也写不出来)。

注意:抱歉没有实际的图像,只有一个有效的链接,显然这是因为我信誉不足而导致的防垃圾邮件功能。

2 个回答

2

我写了这个工具(用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();
}
1

你有没有想过为什么不能直接使用 gtk.StatusIcon.set_from_file 呢?这个方法对透明图片效果很好。

至于你的第二个问题,我很怀疑在 gtk.StatusIcon 上使用 pango 是可能的。
你为什么需要在图标上写文字呢?

撰写回答