开罗的背景和持久性?

时间:2010-08-18 14:32:07

标签: python gtk pygtk cairo pycairo

我刚刚开始使用pycairo,我遇到了以下有趣的错误。我编写的程序创建了一个简单的gtk窗口,在其上绘制一个矩形,然后有一个回调来在任何类型的键盘输入上绘制一个随机线。但是,似乎每个键盘输入,我必须创建一个新的上下文,或者在程序收到第一个键盘输入时(特别是在.stroke()行上)我得到一个错误。如果重要,错误如下。 'BadDrawable(无效的Pixmap或Window参数)'。 (详细信息:serial 230 error_code 9 request_code 53 minor_code 0)

#! /usr/bin/env python
import pygtk
pygtk.require('2.0')
import gtk, gobject, cairo, math, random
# Create a GTK+ widget on which we will draw using Cairo
class Screen(gtk.DrawingArea):
# Draw in response to an expose-event
  __gsignals__ = { "expose-event": "override" }

  # Handle the expose-event by drawing
  def do_expose_event(self, event):
    # Create the cairo context
    self.cr = self.window.cairo_create()
    # Restrict Cairo to the exposed area; avoid extra work
    self.cr.rectangle(event.area.x, event.area.y, event.area.width, event.area.height)
    self.cr.clip()

    self.draw(*self.window.get_size())

  def key_press_event(self, *args):
    # print args
    self.cr = self.window.cairo_create() # This is the line I have to add
    # in order to make this function not throw the error. Note that cr is only
    # given as attribute of self in order to stop it going out of scope when this line
    # doesn't exist
    self.cr.set_source_rgb(random.random(), random.random(), random.random())
    self.cr.move_to(*[z/2.0 for z in self.window.get_size()])
    self.cr.line_to(*[z*random.random() for z in self.window.get_size()])
    self.cr.stroke()

  def draw(self, width, height):
    # Fill the background with gray
    self.cr.set_source_rgb(.5,.5,.5)
    self.cr.rectangle(0, 0, width,height)
    self.cr.fill()

    self.cr.set_source_rgb(1,0,0)
    self.cr.arc(width/2.0, height/2.0, min(width,height)/2.0 - 20.0, 0.0, 2.0*math.pi)
    self.cr.stroke()

#create a gtk window, attach to exit button, and whatever is passed as arg becomes the body of the window. AWESOME
def run(Widget):
  window = gtk.Window()
  widget = Widget()
  window.connect("delete-event", gtk.main_quit)
  window.connect('key-press-event',widget.key_press_event)
  widget.show()
  window.add(widget)
  window.present()
  gtk.main()

if __name__ == "__main__":
  run(Screen)

感谢您的帮助!

(更新:我正在玩,我意识到以下情况:当我调整窗口大小时,添加的所有新对象都会被删除(或者至少不再出现?)

3 个答案:

答案 0 :(得分:2)

开罗的图纸根本不存在。 (最好不要将它们视为“对象” - 它不像画布库那样你可以移动它们或在绘制它们之后转换它们。)你必须在曝光处理程序中完成所有绘图,或者它正如你所发现的那样,只要重新绘制窗口,它就会消失。

由于双缓冲,cairo上下文不会持续存在:请参阅note in the C documentation,遗憾的是我在PyGTK文档中找不到任何地方。

在上面的代码中,您应该在keypress处理程序中生成随机行的坐标和颜色,并将它们保存在数组中。然后在公开处理程序中,按顺序绘制数组中的每一行。

答案 1 :(得分:1)

虽然您必须在每次运行时创建上下文,但您可以通过禁用小部件的双缓冲来实现您正在寻找的持久性。

这是一个使用仓鼠图形库的例子:

https://github.com/projecthamster/experiments/blob/master/many_lines.py

答案 2 :(得分:0)

讨论的许多方面的持久性:

某些曲面上的图形不会持久存在:GUI曲面。您应该在公开回调中重绘它们。

不应将PyCairo对象视为持久对象,仅作为C语言中Cairo库函数的接口。

开罗上下文的内容(路径和填充)不会超出stroke()或fill()操作。

GUI表面的上下文在曝光事件之间不会持续存在(因为双缓冲?)(我不知道上下文是否会持续存在于其他表面即设备。)所以你不能使用cairo context来存储视口的属性(窗口到文档上,即用户坐标中的模型。)

视觉持久性是人眼在停止后看到光的趋势。幽灵和闪烁是它在动画或视频中的症状。禁用双缓冲可让您在绘制时看到事物,即在一个曝光事件中启用动画(模拟视觉持久性的症状。)禁用双缓冲不会使GUI表面上的上下文在公开事件之间持久存在。

记忆的持久性是你真实的持久性,或者我应该说超现实主义。