原文: https://zetcode.com/gfx/pycairo/clipmask/

在 PyCairo 教程的这一部分中,我们将讨论剪切和遮罩操作。

剪裁

剪裁是将绘图限制为特定区域。 这样做是出于效率方面的考虑,并会产生有趣的效果。 PyCairo 具有clip()方法来设置裁剪。

  1. #!/usr/bin/python
  2. '''
  3. ZetCode PyCairo tutorial
  4. This program shows how to perform
  5. clipping in PyCairo.
  6. author: Jan Bodnar
  7. website: zetcode.com
  8. last edited: August 2012
  9. '''
  10. from gi.repository import Gtk, GLib
  11. import cairo
  12. import math
  13. import random
  14. class Example(Gtk.Window):
  15. def __init__(self):
  16. super(Example, self).__init__()
  17. self.init_ui()
  18. self.load_image()
  19. self.init_vars()
  20. def init_ui(self):
  21. self.darea = Gtk.DrawingArea()
  22. self.darea.connect("draw", self.on_draw)
  23. self.add(self.darea)
  24. GLib.timeout_add(100, self.on_timer)
  25. self.set_title("Clipping")
  26. self.resize(300, 200)
  27. self.set_position(Gtk.WindowPosition.CENTER)
  28. self.connect("delete-event", Gtk.main_quit)
  29. self.show_all()
  30. def load_image(self):
  31. self.image = cairo.ImageSurface.create_from_png("beckov.png")
  32. def init_vars(self):
  33. self.pos_x = 128
  34. self.pos_y = 128
  35. self.radius = 40
  36. self.delta = [3, 3]
  37. def on_timer(self):
  38. self.pos_x += self.delta[0]
  39. self.pos_y += self.delta[1]
  40. self.darea.queue_draw()
  41. return True
  42. def on_draw(self, wid, cr):
  43. w, h = self.get_size()
  44. if (self.pos_x < 0 + self.radius):
  45. self.delta[0] = random.randint(5, 9)
  46. elif (self.pos_x > w - self.radius):
  47. self.delta[0] = -random.randint(5, 9)
  48. if (self.pos_y < 0 + self.radius):
  49. self.delta[1] = random.randint(5, 9)
  50. elif (self.pos_y > h - self.radius):
  51. self.delta[1] = -random.randint(5, 9)
  52. cr.set_source_surface(self.image, 1, 1)
  53. cr.arc(self.pos_x, self.pos_y, self.radius, 0, 2*math.pi)
  54. cr.clip()
  55. cr.paint()
  56. def main():
  57. app = Example()
  58. Gtk.main()
  59. if __name__ == "__main__":
  60. main()

在此示例中,我们将裁剪图像。 圆圈在窗口区域上移动并显示基础图像的一部分。 这就像我们从孔中看一样。

  1. def load_image(self):
  2. self.image = cairo.ImageSurface.create_from_png("beckov.png")

这是基础图像。 每个计时器周期,我们都会看到此图像的一部分。

  1. if (self.pos_x < 0 + self.radius):
  2. self.delta[0] = random.randint(5, 9)
  3. elif (self.pos_x > w - self.radius):
  4. self.delta[0]= -random.randint(5, 9)

如果圆碰到窗口的左侧或右侧,则圆的移动方向会随机变化。 顶部和底部也一样。

  1. cr.arc(self.pos_x, self.pos_y, self.radius, 0, 2*math.pi)

此行为 Cairo 上下文添加了一条循环路径。

  1. cr.clip()

clip()设置剪切区域。 裁剪区域是当前使用的路径。 当前路径是通过arc()方法调用创建的。

  1. cr.paint()

paint()在当前剪裁区域内的任何地方绘制当前源。

PyCairo 剪裁&遮罩 - 图1

图:剪裁

遮罩

在将源应用于表面之前,先对其进行过滤。 遮罩用作过滤器。 遮罩确定在哪里应用源,在哪里不应用。 遮罩的不透明部分允许复制源。 透明零件不允许将源复制到表面。

  1. #!/usr/bin/python
  2. '''
  3. ZetCode PyCairo tutorial
  4. This program demonstrates masking.
  5. author: Jan Bodnar
  6. website: zetcode.com
  7. last edited: August 2012
  8. '''
  9. from gi.repository import Gtk
  10. import cairo
  11. class Example(Gtk.Window):
  12. def __init__(self):
  13. super(Example, self).__init__()
  14. self.init_ui()
  15. self.load_image()
  16. def init_ui(self):
  17. darea = Gtk.DrawingArea()
  18. darea.connect("draw", self.on_draw)
  19. self.add(darea)
  20. self.set_title("Masking")
  21. self.resize(310, 100)
  22. self.set_position(Gtk.WindowPosition.CENTER)
  23. self.connect("delete-event", Gtk.main_quit)
  24. self.show_all()
  25. def load_image(self):
  26. self.ims = cairo.ImageSurface.create_from_png("omen.png")
  27. def on_draw(self, wid, cr):
  28. cr.mask_surface(self.ims, 0, 0);
  29. cr.fill()
  30. def main():
  31. app = Example()
  32. Gtk.main()
  33. if __name__ == "__main__":
  34. main()

在该示例中,遮罩确定在哪里绘画和在哪里不绘画。

  1. cr.mask_surface(self.ims, 0, 0);
  2. cr.fill()

我们使用图像作为遮罩,从而将其显示在窗口上。

PyCairo 剪裁&遮罩 - 图2

图:遮罩

蒙蔽效果

在此代码示例中,我们将忽略图像。 这类似于我们使用卷帘所做的。

  1. #!/usr/bin/python
  2. '''
  3. ZetCode PyCairo tutorial
  4. This program creates a blind down
  5. effect using masking operation.
  6. author: Jan Bodnar
  7. website: zetcode.com
  8. last edited: August 2012
  9. '''
  10. from gi.repository import Gtk, GLib
  11. import cairo
  12. import math
  13. class Example(Gtk.Window):
  14. def __init__(self):
  15. super(Example, self).__init__()
  16. self.init_ui()
  17. self.load_image()
  18. self.init_vars()
  19. def init_ui(self):
  20. self.darea = Gtk.DrawingArea()
  21. self.darea.connect("draw", self.on_draw)
  22. self.add(self.darea)
  23. GLib.timeout_add(35, self.on_timer)
  24. self.set_title("Blind down")
  25. self.resize(325, 250)
  26. self.set_position(Gtk.WindowPosition.CENTER)
  27. self.connect("delete-event", Gtk.main_quit)
  28. self.show_all()
  29. def load_image(self):
  30. self.image = cairo.ImageSurface.create_from_png("beckov.png")
  31. def init_vars(self):
  32. self.timer = True
  33. self.h = 0
  34. self.iw = self.image.get_width()
  35. self.ih = self.image.get_height()
  36. self.ims = cairo.ImageSurface(cairo.FORMAT_ARGB32,
  37. self.iw, self.ih)
  38. def on_timer(self):
  39. if (not self.timer):
  40. return False
  41. self.darea.queue_draw()
  42. return True
  43. def on_draw(self, wid, cr):
  44. ic = cairo.Context(self.ims)
  45. ic.rectangle(0, 0, self.iw, self.h)
  46. ic.fill()
  47. self.h += 1
  48. if (self.h == self.ih):
  49. self.timer = False
  50. cr.set_source_surface(self.image, 10, 10)
  51. cr.mask_surface(self.ims, 10, 10)
  52. def main():
  53. app = Example()
  54. Gtk.main()
  55. if __name__ == "__main__":
  56. main()

盲目效应背后的想法很简单。 图像高度为 h 像素。 我们绘制高度为 1px 的 0、1、2 … 线。 每个周期,图像的一部分高 1px,直到整个图像可见为止。

  1. def load_image(self):
  2. self.image = cairo.ImageSurface.create_from_png("beckov.png")

load_image()方法中,我们从 PNG 图像创建图像表面。

  1. def init_vars(self):
  2. self.timer = True
  3. self.h = 0
  4. self.iw = self.image.get_width()
  5. self.ih = self.image.get_height()
  6. self.ims = cairo.ImageSurface(cairo.FORMAT_ARGB32,
  7. self.iw, self.ih)

init_vars()方法中,我们初始化一些变量。 我们启动self.timerself.h变量。 我们得到加载图像的宽度和高度。 然后我们创建一个空的图像表面。 它将用我们之前创建的图像表面的像素线填充。

  1. ic = cairo.Context(self.ims)

我们从空图像源创建一个 cairo 上下文。

  1. ic.rectangle(0, 0, self.iw, self.h)
  2. ic.fill()

我们在最初为空的图像中绘制一个矩形。 矩形每个周期将高出 1 像素。 以这种方式创建的图像稍后将用作遮罩。

  1. self.h += 1

要显示的图像高度增加一个单位。

  1. if (self.h == self.ih):
  2. self.timer = False

当我们在 GTK 窗口上绘制整个图像时,我们将停止timer方法。

  1. cr.set_source_surface(self.image, 10, 10)
  2. cr.mask_surface(self.ims, 10, 10)

城堡的图像被设置为绘画的来源。 mask_surface()使用表面的 Alpha 通道作为遮罩来绘制电流源。

本章介绍了 PyCairo 中的剪裁和遮罩。