PyCairo 中的剪裁和屏蔽

2024-04-20 20:58
文章标签 剪裁 屏蔽 pycairo

本文主要是介绍PyCairo 中的剪裁和屏蔽,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在 PyCairo 教程的这个部分,我们将讨论剪裁和屏蔽操作。

剪裁

裁剪 是将绘制限定在某一区域内。这样做有一些效率的因素,或者为了创建有趣的效果。PyCairo 有一个 clip() 方法用于设置裁剪区域。

#!/usr/bin/python'''
ZetCode PyCairo tutorialThis program shows how to perform
clipping in PyCairo.author: Jan Bodnar
website: zetcode.com
last edited: August 2012
'''import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gtk, GLib
import cairo
import math
import randomclass Example(Gtk.Window):def __init__(self):super(Example, self).__init__()self.init_ui()self.load_image()self.init_vars()def init_ui(self):self.darea = Gtk.DrawingArea()self.darea.connect("draw", self.on_draw)self.add(self.darea)GLib.timeout_add(100, self.on_timer)self.set_title("Clipping")self.resize(300, 200)self.set_position(Gtk.WindowPosition.CENTER)self.connect("delete-event", Gtk.main_quit)self.show_all()def load_image(self):self.image = cairo.ImageSurface.create_from_png("beckov.png")def init_vars(self):self.pos_x = 128self.pos_y = 128self.radius = 40self.delta = [3, 3]def on_timer(self):self.pos_x += self.delta[0]self.pos_y += self.delta[1]self.darea.queue_draw()return Truedef on_draw(self, wid, cr):w, h = self.get_size()if (self.pos_x < 0 + self.radius):self.delta[0] = random.randint(5, 9)elif (self.pos_x > w - self.radius):self.delta[0] = -random.randint(5, 9)if (self.pos_y < 0 + self.radius):self.delta[1] = random.randint(5, 9)elif (self.pos_y > h - self.radius):self.delta[1] = -random.randint(5, 9)cr.set_source_surface(self.image, 1, 1)cr.arc(self.pos_x, self.pos_y, self.radius, 0, 2 * math.pi)cr.clip()cr.paint()def main():app = Example()Gtk.main()if __name__ == "__main__":main()

在这个例子中,我们将裁剪一幅图片。一个圆圈在窗口区域移动,并显示下面的图片的一部分。这就好像我们通过一个洞看过去一样。

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

这是下面的图片。每一个定时器周期,我们将看到这幅图片的一部分。

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

如果圆圈击中了窗口的左边或右边,则圆圈移动的方向会随机地改变。对于上边和下边也一样。

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

这一行给 Cairo 上下文添加一个圆形的 Path。

        cr.clip()

clip() 设置裁剪区域。裁剪区域是当前正在使用的 Path。当前的 path 由 arc() 方法调用创建。

        cr.paint()

paint() 用当前的 source 描绘当前裁剪区域内的部分。

图:剪裁

屏蔽

在 source 被应用于 surface 之前,它首先会被过滤。mask 被用作一个过滤器。mask 决定 source 的哪个部分被应用,而哪个部分不会。mask 不透明的部分允许复制 source。透明的部分则不允许复制 source 到 surface。

#!/usr/bin/python'''
ZetCode PyCairo tutorialThis program demonstrates masking.author: Jan Bodnar
website: zetcode.com
last edited: August 2012
'''import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gtk
import cairoclass Example(Gtk.Window):def __init__(self):super(Example, self).__init__()self.init_ui()self.load_image()def init_ui(self):darea = Gtk.DrawingArea()darea.connect("draw", self.on_draw)self.add(darea)self.set_title("Masking")self.resize(310, 100)self.set_position(Gtk.WindowPosition.CENTER)self.connect("delete-event", Gtk.main_quit)self.show_all()def load_image(self):self.ims = cairo.ImageSurface.create_from_png("omen.png")def on_draw(self, wid, cr):cr.mask_surface(self.ims, 0, 0);cr.fill()def main():app = Example()Gtk.main()if __name__ == "__main__":main()

在这个例子中,屏蔽决定了哪些地方需要绘制哪些地方不绘制。

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

我们使用一幅图片作为 mask,这将会把它显示在窗口中。

图:屏蔽

Blind down 效果

在这个代码例子中,我们将 blind down 我们的图片。这类似于我们使用的遮光窗帘。

#!/usr/bin/python'''
ZetCode PyCairo tutorialThis program creates a blind down
effect using masking operation.author: Jan Bodnar
website: zetcode.com
last edited: August 2012
'''import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gtk, GLib
import cairo
import mathclass Example(Gtk.Window):def __init__(self):super(Example, self).__init__()self.init_ui()self.load_image()self.init_vars()def init_ui(self):self.darea = Gtk.DrawingArea()self.darea.connect("draw", self.on_draw)self.add(self.darea)GLib.timeout_add(35, self.on_timer)self.set_title("Blind down")self.resize(325, 250)self.set_position(Gtk.WindowPosition.CENTER)self.connect("delete-event", Gtk.main_quit)self.show_all()def load_image(self):self.image = cairo.ImageSurface.create_from_png("beckov.png")def init_vars(self):self.timer = Trueself.h = 0self.iw = self.image.get_width()self.ih = self.image.get_height()self.ims = cairo.ImageSurface(cairo.FORMAT_ARGB32,self.iw, self.ih)def on_timer(self):if (not self.timer):return Falseself.darea.queue_draw()return Truedef on_draw(self, wid, cr):ic = cairo.Context(self.ims)ic.rectangle(0, 0, self.iw, self.h)ic.fill()self.h += 1if (self.h == self.ih):self.timer = Falsecr.set_source_surface(self.image, 10, 10)cr.mask_surface(self.ims, 10, 10)def main():app = Example()Gtk.main()if __name__ == "__main__":main()

blend down 效果背后的想法相当的简单。图像有 h 个像素高。我们绘制 0,1, 2… 个 1 像素高的行。每个周期,图像的部分多出一像素的高度,直到整幅图片都变得可见为止。

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

load_image() 方法中,我们由一幅 PNG 图片创建一个图片 surface。

    def init_vars(self):self.timer = Trueself.h = 0self.iw = self.image.get_width()self.ih = self.image.get_height()self.ims = cairo.ImageSurface(cairo.FORMAT_ARGB32,self.iw, self.ih)

init_vars() 方法中,我们初始化一些变量。我们初始化 self.timerself.h 变量。我们获取所加载的图片的宽度和高度。然后我们创建一个空的图像 surface。它将会被来自于先前我们所创建的图像 surface 的像素行所填充。

        ic = cairo.Context(self.ims)

我们由空的图像 source 创建一个 cairo 上下文。

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

我们向最初为空的图像中绘制一个矩形。矩形每个周期高出 1px。用这种方式创建的图像将在后面作为一个 mask。

        self.h += 1

将要显示的图像的高度被加了一个单元。

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

当我们在 GTK 窗口中绘制了整个图片时,我们停掉定时器方法。

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

城堡的图片被设置为绘制时的 source。mask_surface() 绘制当前的 source,使用 surface 的 alpha 通道作为一个 mask。

图:Blind down

Blind down 是一种动画效果。

本章讨论了PyCairo中的裁剪和屏蔽。

原文

Done.

这篇关于PyCairo 中的剪裁和屏蔽的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/921308

相关文章

Linux中如何屏蔽信号

本篇文章主要学习Linux的信号处理机制,着重学习屏蔽信号部分。屏蔽信号处理的两种方式类似于信号的捕获,一种方式是直接对其设置,另一种方式是先获得描述符的掩码,然后对其设置操作。 本文主要参考自《嵌入式linux系统使用开发》,作者何永琪,Thanks. 在linux系统中,如何处理某个进程发送的一个特定信号呢?一般来说有三种方式: 1) 忽略信号 2) 屏蔽信号 3) 为该信号添

CF #283 (Div. 2) A.(屏蔽数组元素)

题目链接:http://codeforces.com/contest/496/problem/A 解题思路: n不是很大,所以暴力。每次屏蔽掉a[ i ]中的一个元素,注意头和尾不能屏蔽。屏蔽后当i == j 时做特殊处理,即cnt = a[ i+ 1 ] - a[ i - 1 ]。最后更新最小值即可。 完整代码: #include <functional>#includ

Android扫描名片的动画以及剪裁扫描框的图片

参考代码:https://github.com/smartown/CertificateCamera 本文章在上述参考代码的基础上进行修改,用到的SurfaceView是链接代码里的CameraPreview,不过本人比较懒,直接加入相机连续对焦模式代替点击屏幕对焦   ,在CameraPreview中添加 parameters.setFocusMode(Camera.Parameters.F

Scrapy ——如何防止被ban 屏蔽 之策略大集合(六)

话说在尝试设置download_delay小于1,并且无任何其他防止被ban的策略之后,我终于成功的被ban了。   关于scrapy的使用可参见之前文章:   http://blog.csdn.net/u012150179/article/details/34913315 http://blog.csdn.net/u012150179/article/details/34486677

ffmpeg h264解码, 屏蔽因为网络丢包等各种原因导致的花屏帧

ffmpeg h264解码, 屏蔽因为网络丢包等各种原因导致的花屏帧  ---->看来问题只能这样解决了,现在还要多测测,防止产生新的问题。目前来看,对现有代码没有影响,花屏的帧直接屏蔽掉了。 思路: 问了下机顶盒方面h264解码的,他们用的是硬件解码,他们做到不花屏的方法就是简单的设置了一个硬件解码提供的接口:设置了错误处理模式。 我想这个错误处理模式肯定对那些错误的帧直接屏

freeswitch借助fail2ban屏蔽骚扰注册

fail2ban屏蔽fs中的骚扰注册 背景Fail2Ban安装配置 背景 在阿里云上部了一套fs满足公网业务需要。布置好后打开fs_cli看了一眼,全是来自公网的骚扰注册,足见公网环境的风险(不过aws似乎情况稍好)。 Fail2Ban 使用Fail2Ban来防御注册骚扰的主要原因是,Fail2Ban已经和freeswitch有了很好的集成。Fail2Ban的工作原

kernel 下串口serial输入输出控制,屏蔽log的输出

最近工作在调试usb虚拟串口,让其作为kernel启动的调试串口,以及user空间的输入输出控制台。 利用这个机会,学习下printk如何选择往哪个console输出以及user空间下控制台如何选择,记录与此,与大家共享,也方便自己以后翻阅。 Kernel版本号:3.4.55 依照我的思路(还是时间顺序)分了4部分,指定kernel调试console ,  kernel下printk con

android屏蔽软键盘并且显示光标

转自: http://blog.csdn.net/danielinbiti/article/details/12972671 如果是android4.0以下,那么 editText.setInputType(InputType.TYPE_NULL); editText.setInputType(InputType.TYPE_NULL);就够了,and

ExtJS 屏蔽backspace

/*** @author wangsr* @date 2012-09-06* 键盘事件管理*/Ext.define("Gboat2.desktop.util.KeyManager", {alternateClassName : "GKeyManager",singleton : true,maskBackspace : function() {var DOC = Ext.getDoc();DOC

SQL server 屏蔽除数为0错误

--屏蔽除数为0错误 SET ANSI_WARNINGS off SET ARITHABORT off SET ARITHIGNORE on 这些 T-SQL 语句控制 SQL Server 的行为,特别是在处理数据和错误时的行为。下面是每个设置的具体作用: SET NOCOUNT ON: 作用: 当执行 SQL 语句时,SQL Server 通常会在每条命令执行后返回一条消息,指