ZetCode

根窗口

最后修改于 2023 年 7 月 17 日

在本篇 PyCairo 教程中,我们将使用根窗口。根窗口是我们通常放置图标快捷方式的桌面窗口。

可以操作根窗口。从程序员的角度来看,它只是一个特殊的窗口。

透明窗口

我们的第一个示例创建一个透明窗口。我们可以看到窗口对象下方的图像。

#!/usr/bin/python

'''
ZetCode PyCairo tutorial 

This code example shows how to
create a transparent window.

author: Jan Bodnar
website: zetcode.com
'''

from gi.repository import Gtk
import cairo


class Example(Gtk.Window):

    def __init__(self):
        super(Example, self).__init__()
        
        self.tran_setup()
        self.init_ui()
        
        
    def init_ui(self):    

        self.connect("draw", self.on_draw)        

        self.set_title("Transparent window")
        self.resize(300, 250)
        self.set_position(Gtk.WindowPosition.CENTER)
        self.connect("delete-event", Gtk.main_quit)
        self.show_all()
        
        
    def tran_setup(self):    
        
        self.set_app_paintable(True)  
        screen = self.get_screen()
        
        visual = screen.get_rgba_visual()       
        if visual != None and screen.is_composited():
            self.set_visual(visual)              
        
    
    def on_draw(self, wid, cr):

        cr.set_source_rgba(0.2, 0.2, 0.2, 0.4)
        cr.set_operator(cairo.OPERATOR_SOURCE)
        cr.paint()
       
    
def main():
    
    app = Example()
    Gtk.main()
        
        
if __name__ == "__main__":    
    main()

要创建透明窗口,我们需要获取屏幕对象的 visual,并将其设置给我们的窗口。在 on_draw 方法中,我们在屏幕的 visual 对象上进行绘制。这创造了一种部分透明的错觉。

self.set_app_paintable(True)  

我们必须设置应用程序进行绘制。

screen = self.get_screen()

get_screen 方法返回屏幕对象。

visual = screen.get_rgba_visual()

从屏幕窗口,我们获取其 visual。visual 包含低级显示信息。

if visual != None and screen.is_composited():
    self.set_visual(visual) 

并非所有显示都支持此操作。因此,我们检查我们的屏幕是否支持组合,以及返回的 visual 是否不为 None。我们将屏幕的 visual 设置为我们窗口的 visual。

def on_draw(self, wid, cr):

    cr.set_source_rgba(0.2, 0.2, 0.2, 0.4)
    cr.set_operator(cairo.OPERATOR_SOURCE)
    cr.paint()

我们使用一个部分透明的源在屏幕窗口上绘制。cairo.OPERATOR_SOURCE 创建一个组合操作,我们在源上进行绘制。源就是屏幕窗口。要获得完全透明,我们将 alpha 值设置为 0 或使用 cairo.OPERATOR_CLEAR 操作符。

Transparent window
图:透明窗口

截屏

根窗口对于截屏也很重要。

#!/usr/bin/python

'''
ZetCode PyCairo tutorial 

This code example takes a screenshot.

author: Jan Bodnar
website: zetcode.com
'''

from gi.repository import Gdk
import cairo


def main():
    
    root_win = Gdk.get_default_root_window()

    width = root_win.get_width()
    height = root_win.get_height()    
    
    ims = cairo.ImageSurface(cairo.FORMAT_ARGB32, width, height)                
    pb = Gdk.pixbuf_get_from_window(root_win, 0, 0, width, height)
        
    cr = cairo.Context(ims)    
    Gdk.cairo_set_source_pixbuf(cr, pb, 0, 0)     
    cr.paint()

    ims.write_to_png("screenshot.png")
        
        
if __name__ == "__main__":    
    main()

该示例捕获整个屏幕的快照。

root_win = Gdk.get_default_root_window()

我们通过调用 Gdk.get_default_root_window 方法来获取根窗口。

width = root_win.get_width()
height = root_win.get_height() 

我们确定根窗口的宽度和高度。

ims = cairo.ImageSurface(cairo.FORMAT_ARGB32, width, height)

创建一个空的图像表面。它的尺寸与根窗口相同。

pb = Gdk.pixbuf_get_from_window(root_win, 0, 0, width, height)

我们使用 Gdk.pixbuf_get_from_window 方法调用从根窗口获取一个 pixbuf。pixbuf 是一个描述内存中图像的对象。它被 GTK 库使用。

cr = cairo.Context(ims)    
Gdk.cairo_set_source_pixbuf(cr, pb, 0, 0)     
cr.paint()

在上面的代码行中,我们在之前创建的图像表面上创建一个 Cairo 绘图上下文。我们将 pixbuf 放置在绘图上下文中,并将其绘制到表面上。

ims.write_to_png("screenshot.png")

使用 write_to_png 方法将图像表面写入 PNG 图像。

显示消息

在第三个示例中,我们在桌面窗口上显示一条消息。

#!/usr/bin/python

'''
ZetCode PyCairo tutorial 

This code example shows a message on the desktop
window.

author: Jan Bodnar
website: zetcode.com
'''


from gi.repository import Gtk, Gdk, Pango
import cairo


class Example(Gtk.Window):

    def __init__(self):
        super(Example, self).__init__()
        
        self.setup()       
        self.init_ui()

        
    def setup(self):    
        
        self.set_app_paintable(True)   
        self.set_type_hint(Gdk.WindowTypeHint.DOCK)
        self.set_keep_below(True)
        
        screen = self.get_screen()
        visual = screen.get_rgba_visual()       
        if visual != None and screen.is_composited():
            self.set_visual(visual)          
        
        
    def init_ui(self):    

        self.connect("draw", self.on_draw)        
        
        lbl = Gtk.Label()
        text = "ZetCode, tutorials for programmers."
        lbl.set_text(text)        
        
        fd = Pango.FontDescription("Serif 20")
        lbl.modify_font(fd)                
        lbl.modify_fg(Gtk.StateFlags.NORMAL,Gdk.color_parse("white"))        
        
        self.add(lbl)

        self.resize(300, 250)
        self.set_position(Gtk.WindowPosition.CENTER)
        self.connect("delete-event", Gtk.main_quit)
        self.show_all()
                                      
    
    def on_draw(self, wid, cr):
        
        cr.set_operator(cairo.OPERATOR_CLEAR)
        cr.paint()
        cr.set_operator(cairo.OPERATOR_OVER)
        
    
def main():
        
        app = Example()
        Gtk.main()        

        
if __name__ == "__main__":    
    import signal
    signal.signal(signal.SIGINT, signal.SIG_DFL)
    main()

代码在根窗口上显示一个消息标签。

self.set_app_paintable(True) 

我们将操作应用程序窗口,因此我们使其可绘制。

self.set_type_hint(Gdk.WindowTypeHint.DOCK)

实现此窗口提示可以移除窗口边框和装饰。

self.set_keep_below(True) 

我们让应用程序始终保持在底部,正好在根窗口之上。

screen = self.get_screen()
visual = screen.get_rgba_visual()       
if visual != None and screen.is_composited():
    self.set_visual(visual)  

我们将屏幕的 visual 设置为我们应用程序的 visual。

lbl = Gtk.Label()
text = "ZetCode, tutorials for programmers."
lbl.set_text(text)    

我们将一个消息标签放在应用程序窗口上。

fd = Pango.FontDescription("Serif 20")
lbl.modify_font(fd)                
lbl.modify_fg(Gtk.StateFlags.NORMAL,Gdk.color_parse("white"))   

借助 Pango 模块,我们更改文本的外观。

def on_draw(self, wid, cr):
    
    cr.set_operator(cairo.OPERATOR_CLEAR)
    cr.paint()
    cr.set_operator(cairo.OPERATOR_OVER)

我们使用 cairo.OPERATOR_CLEAR 操作符清除窗口的背景。然后我们设置 cairo.OPERATOR_CLEAR 以便绘制标签小部件。

if __name__ == "__main__":    
    import signal
    signal.signal(signal.SIGINT, signal.SIG_DFL)
    main()

有一个旧的 bug,它不允许我们使用 Ctrl+C 快捷方式终止从终端启动的应用程序。添加这两行是对此的一个解决方法。

Message on the root window
图:根窗口上的消息

在本章中,我们使用 PyCairo 处理了桌面窗口。