在Fedora 24上的Python3中使用`cairo.Region`时出现分段错误

时间:2016-07-10 02:27:52

标签: python python-3.x gtk pygobject pycairo

我正在尝试将输入形状应用于窗口,因此我可以使用透明框架,只有可点击的有趣部分等。

我在PyGObject(Python 3)中使用Cairo图形。

Object是一个常规的Gtk Windows对象,然后定义一个Cairo绘图区域:

class Face(Gtk.Window):
    def __init__(self):
        super(Face, self).__init__()
        self.init_ui()

    def init_ui(self):
        [...]
        self.darea = Gtk.DrawingArea()
        self.darea.connect("draw", self.on_draw)
        self.add(self.darea)

    def on_draw(self, widget, cr):
        [... (drawing a couple shapes into the context)]
        sface = cr.get_group_target()
        mregion = Gdk.cairo_region_create_from_surface(sface)
        # the above line produces the error
        # the following lines is wishful thinking at this point...
        #self.get_window().input_shape_combine_region(mregion, 0, 0)

因此,每当必须重绘绘制区域时,都会调用函数on_draw(),即窗口移动,调整大小,隐藏后显示等等。

大多数窗口实际上是空的,因为它应该是一个大部分半透明的框架,只有可见部分应该是可点击的。 但是,我收到以下错误:

Traceback (most recent call last):
  File "./lsc.py", line 236, in on_draw
    mregion = Gdk.cairo_region_create_from_surface(sface)
TypeError: Couldn't find foreign struct converter for 'cairo.Region'
python3: cairo-surface.c:953: cairo_surface_destroy: Assertion `CAIRO_REFERENCE_COUNT_HAS_REFERENCE (&surface->ref_count)' failed.
zsh: abort (core dumped)  ./lsc.py

安装了Python3,PyGObject和Cairo库的软件包,我还导入了开罗的两个部分:

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

此时我不完全确定,这是一个Python问题,还是我的错误。 我不确定即使使用cairo.Region甚至是这样的,我正在进行的例子是:http://www.programcreek.com/python/example/81481/cairo.Region

修改

此时我很困惑发生了什么事。我一直在寻找这个,这是我发现的:

PyGI和开罗似乎存在某种错误。

当我只使用cairo的内省部分时:

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

然后运行我的脚本,我收到此错误:

raceback (most recent call last):
  File "./lsc.py", line 164, in on_draw
    cr.set_operator(cairo.OPERATOR_SOURCE)
  File "/usr/lib64/python3.5/site-packages/gi/module.py", line 139, in __getattr__
    self.__name__, name))
AttributeError: 'gi.repository.cairo' object has no attribute 'OPERATOR_SOURCE'

显然我在代码中使用cairo.OPERATOR_SOURCE绘制到cairo上下文。然而,内省包甚至似乎不包含用于在表面上绘制的te运算符。很奇怪。

当我使用 非内省模块时:

from gi.repository import Gtk, Gdk
import cairo

我得到了与问题第一部分相同的错误。

完成后,这是我当前安装的cairo包列表:

cairo-devel-1.14.6-1.fc24.x86_64
pycairo-devel-1.10.0-4.fc24.x86_64
cairo-gobject-1.14.6-1.fc24.i686
mingw32-cairo-1.14.6-1.fc24.noarch
python3-cairocffi-0.7.2-5.fc24.noarch
cairo-1.14.6-1.fc24.i686
cairo-1.14.6-1.fc24.x86_64
pycairo-1.10.0-4.fc24.x86_64
python3-cairosvg-1.0.19-3.fc24.noarch
cairomm-devel-1.12.0-2.fc24.x86_64
cairo-clock-0.3.4-17.fc24.x86_64
cairomm-1.12.0-2.fc24.x86_64
cairo-gobject-1.14.6-1.fc24.x86_64
python3-cairo-1.10.0-15.fc24.x86_64
mingw32-cairomm-1.12.0-2.fc24.noarch
python3-cairo-devel-1.10.0-15.fc24.x86_64
cairo-gobject-devel-1.14.6-1.fc24.x86_64

我想你会发现,所有这些都是最近的,并且符合我上面链接的示例中所述的要求。

另外,请考虑countdown.py的第268-274行:

# make window click-through, this needs pycairo 1.10.0 for python3
# to work
rect = cairo.RectangleInt (0, 0, 1, 1)
region = cairo.Region (rect)
if (not region.is_empty ()):
    self.window.input_shape_combine_region (None)
    self.window.input_shape_combine_region (region)

作者没有使用内省的cairo,他只是import cairo,这让我相信它可能是更正确的方式。

无论如何,整个脚本也不起作用。

此时我又被卡住了。我似乎无法找到如何使用此功能的最新和/或工作示例。

到目前为止,似乎使用内省版cairo是没有用的,因为它几乎没有定义原始元素。但是,在使用常规pycairo时(通过导入import cairo),我得到的错误让我首先发布了这个问题。

我开始相信我有一个概念错误,也许?

1 个答案:

答案 0 :(得分:1)

好的...我知道这篇文章过时了,但是我在寻找相同问题的答案时发现了。 (这是我对堆栈溢出的第一个贡献,请耐心等待我)。我想在GTK 3中创建一个非矩形窗口,经过数小时的研究,我还是空了。

最终,我找到了与上述Polemon相同的示例程序,并通读了Cairo和Gdk文档。最后,Gdk.cairo_region_create_from_surface()后跟GdkWindow.input_shape_combine_region(),这两个实现创建仅对开罗表面的非透明区域的输入作出响应的非矩形窗口的功能是。 >

现在,my application致力于使用GtkDialog窗口创建应用程序启动器,但是此代码可以应用于标准的GTKWindow。

这是我要实现此效果所遵循的步骤。首先,我在对话框窗口的构造函数中添加了以下几行:

    self.set_decorated(False)  # Creates a borderless window without a title bar
    self.set_app_paintable(True) # Tells GTK not to redraw the window area but let the application handle the drawing
    self.connect('draw', self.draw) # Registers a callback function for the draw signal

接下来,我定义了负责处理绘制信号的回调函数。在其中添加了以下内容:

    # Create a Cairo surface from a PNG image
    self.image = cairo.ImageSurface.create_from_png("image-with-transparent-regions.png")
    # Create a new Cairo region based on this Cairo surface
    region = Gdk.cairo_region_create_from_surface(self.image)
    # Combine this region with the window dialog window region.
    # Input area is now restricted to non-transparent areas of the dialog window 
    self.input_shape_combine_region(region)

    # Sets the Cairo compositing Operator to replace the contents of the window background
    context.set_operator(cairo.OPERATOR_SOURCE)
    # Sets the Cairo surface as the pattern to be drawn on the window background
    context.set_source_surface(self.image, 0, 0)
    # Paints the background image with the Cairo surface pattern
    context.paint()
    # Restores the default state of the Cairo compositing operator to draw the source layer on top of the destination window layer.
    # This prevents the background image from being erased by alpha component of the widgets being drawn on top of it.
    context.set_operator(cairo.OPERATOR_OVER)

对于有兴趣试用现成代码的人,check out my full program