PyGTK透明窗口

时间:2012-05-28 09:45:31

标签: window pygtk transparent cairo pango

我希望窗口透明,但标签在不透明度方面是100%。我怎样才能做到这一点?顺便说一句:我注意到当我升级到Ubuntu 12.04的统一界面时,window.set_opacity的运行方式与它在GNOME上的效果不同,但即使它完成了窗口内的所有内容也会变得透明。

这是我开始使用的代码......

#!/usr/bin/env python

import pygtk
pygtk.require('2.0')
import gtk
import pango
import time

class Clock:

  def __init__(self):
    me = gtk.Window(gtk.WINDOW_TOPLEVEL)
    me.connect("destroy", lambda w: gtk.main_quit())
    me.set_decorated(False)
    me.set_has_frame(False)
    me.set_resizable(False)
    me.set_property('skip-taskbar-hint', True)
    self.label = gtk.Label()
    self.label.modify_font(pango.FontDescription("FreeSerif Bold 50"))
    attr = pango.AttrList()
    fg_color = pango.AttrForeground(65535, 0, 0, 0, 65535)
    attr.insert(fg_color)
    self.label.set_attributes(attr)

    me.add(self.label)
    me.show_all()

  def update(self):
    self.label.set_text(time.strftime('%H:%M:%S'))
    return True

clock = Clock()
gtk.timeout_add(200, clock.update)
gtk.main()

我在askubuntu上找到了this主题,而这正是我所寻找的,但是现在我遇到了数字时钟显示的问题。任何帮助将不胜感激。

这是我的代码。

#!/usr/bin/env python

import pygtk
pygtk.require('2.0')
import gtk
import pango
import time
import cairo

class Clock (gtk.Window):
  def __init__(self):
    super(Clock, self).__init__()
    self.connect("destroy", lambda w: gtk.main_quit())
    self.set_decorated(False)
    self.set_has_frame(False)
    self.set_resizable(False)
    self.set_property('skip-taskbar-hint', True)
    self.label = gtk.Label()
    self.label.modify_font(pango.FontDescription("FreeSerif Bold 50"))
    attr = pango.AttrList()
    fg_color = pango.AttrForeground(65535, 0, 0, 0, 65535)
    attr.insert(fg_color)
    self.label.set_attributes(attr)

    self.screen = self.get_screen()
    self.visual = self.screen.get_rgba_visual()
    self.set_visual(self.visual)
    self.set_app_paintable(True)
    self.connect("draw", self.area_draw)

    self.add(self.label)
    self.show_all()

  def update(self):
    self.label.set_text(time.strftime('%H:%M:%S'))
    return True

  def area_draw(self, widget, cr):
    cr.set_source_rgba(.2, .2, .2, 0.5)
    cr.set_operator(cairo.OPERATOR_SOURCE)
    cr.paint()
    cr.set_operator(cairo.OPERATOR_OVER)

clock = Clock()
gtk.timeout_add(200, clock.update)
gtk.main()

1 个答案:

答案 0 :(得分:3)

嗯,你必须破解很多这样的标签。 最好的是重新发明轮子:做一个圆形。

绘制自己的标签。 获取窗口上下文(mywin.window.cairo_create()))并跟踪x,y。

def draw(*args):
    ctx = win.window.cairo_create()
    ctx.set_source_rgba(0, 0, 0, 0)
    ctx.set_operator(0) # OPERATOR_CLEAR
    ctx.paint()
    ctx.set_source_rgba(0, .6, 1, .3)
    ctx.arc(w/2, h/2, w/2, 0, 2*3.1415)
    ctx.fill()
    ctx.set_source_rgba(1, .8, 0, 1)
    ctx.show_text('29-May-1234') #you have some work for font, style and so on

谷歌给你一个使用ctypes从文件加载ttf字体的hack;它在Linux上运行得很好(我不知道win)。 注意:上面假设你有一个复合管理器,或者...... rgba for widget是None。 还放了一个

colormap = self.win.get_screen().get_rgba_colormap()
if colormap == None: colormap = self.win.get_screen().get_rgb_colormap()
gtk.widget_set_default_colormap(colormap)

设置rgba如果可行,也值得检查:

if self.win.is_composited():print ('OK!')
else:self.bg = self.capt_screen()

...........

def capt_screen(self):
    x, y = self.win.get_position()
    win = gtk.gdk.get_default_root_window()
    w, h = win.get_size()
    pb = gtk.gdk.Pixbuf(0 , False, 8, w, h)
    self.win.hide()
    pb = pb.get_from_drawable(win, win.get_colormap(), 0, 0, 0, 0, w, h)
    self.win.show_all()
    if (pb != None):
        im = Image.fromstring('RGB', (w, h), pb.get_pixels())
        im = im.crop((x, y, x + self._w, y + self._h))
        im = im.convert('RGBA') 
        return im #or pb

以上是“xp”透明效果:用你的胜利复制bg和bledit。只有当它不是一个复合管理器运行时。对于小部件是好的 - 留在桌面上,但为其他东西flickr:在每个刷新窗口都必须隐藏自己,捕获gb,绘图和显示。

PS:我有一个工作时钟示例,但我使用png。如果你愿意,我可以发邮件给你tar.gz