Pyglet控件的批处理参数batch和分组参数group简析

先来复习一下之前写的两个例程: 

1. 绘制网格线

import pygletwindow = pyglet.window.Window(800, 600)
color = (255, 255, 255, 255)  # 白色
lines = []for y in range(0, window.height, 40):lines.append(pyglet.shapes.Line(0, y, window.width, y, color=color))
for x in range(0, window.width, 40):lines.append(pyglet.shapes.Line(x, 0, x, window.height, color=color))@window.event
def on_draw():window.clear()for line in lines:line.draw()pyglet.app.run()

2. 模拟按钮

import pygletwindow = pyglet.window.Window(800, 600)GRAY = (220, 220, 220, 255)
BLACK = (0, 0, 0, 255)
RED = (255, 0, 0, 255)pattern = pyglet.image.SolidColorImagePattern(color=GRAY)
image = pattern.create_image(100,50)
button1 = pyglet.sprite.Sprite(image, 280, 200)
button2 = pyglet.sprite.Sprite(image, 280+150, 200)
labelb1 = pyglet.text.Label('按钮1', font_size=16, x=305, y=216, color=BLACK)
labelb2 = pyglet.text.Label('按钮2', font_size=16, x=455, y=216, color=BLACK)
message = pyglet.text.Label('', font_size=16, x=20, y=16)
widgets = [message]
widgets.extend([button1, button2, labelb1, labelb2])@window.event
def on_draw():window.clear()for widget in widgets:widget.draw()@window.event
def on_mouse_press(x, y, button, modifier):X, Y, W, H = button1.x, button1.y, button1.width, button1.heightif x in range(X,X+W+1) and y in range(Y,Y+H+1):message.text = '你点击了按钮1'elif x in range(X+150,X+W+151) and y in range(Y,Y+H+1):message.text = '你点击了按钮2'@window.event
def on_mouse_motion(x, y, dx, dy):X, Y, W, H = button1.x, button1.y, button1.width, button1.heightif x in range(X,X+W+1) and y in range(Y,Y+H+1):labelb1.color = REDelif x in range(X+150,X+W+151) and y in range(Y,Y+H+1):labelb2.color = REDelse:labelb1.color = BLACKlabelb2.color = BLACKpyglet.app.run()

以上两个例程,都使用了控件列表,然后在on_draw事件中遍历,基本上实现了“批处理”画图的功能。但是,实现上pyglet库本身就带有这种批处理功能,这就是各个控件都带有的batch参数。看一个pyglet自带的实例:

import pyglet
from pyglet import shapeswindow = pyglet.window.Window(960, 540)
batch = pyglet.graphics.Batch()circle = shapes.Circle(700, 150, 100, color=(50, 225, 30), batch=batch)
square = shapes.Rectangle(200, 200, 200, 200, color=(55, 55, 255), batch=batch)
rectangle = shapes.Rectangle(250, 300, 400, 200, color=(255, 22, 20), batch=batch)
rectangle.opacity = 128
rectangle.rotation = 33
line = shapes.Line(100, 100, 100, 200, width=19, batch=batch)
line2 = shapes.Line(150, 150, 444, 111, width=4, color=(200, 20, 20), batch=batch)
star = shapes.Star(800, 400, 60, 40, num_spikes=20, color=(255, 255, 0), batch=batch)@window.event
def on_draw():window.clear()batch.draw()pyglet.app.run()

是不是更简单,使用了pyglet.graphics.Batch()对象,把需要一起画的控件都装进Batch()里,然后再一起画出来。与以下代码的效果是一样的,但省掉了widgets元组和遍历过程,还是原生的批处理参数batch更简单。

import pyglet
from pyglet import shapeswindow = pyglet.window.Window(960, 540)circle = shapes.Circle(700, 150, 100, color=(50, 225, 30))
square = shapes.Rectangle(200, 200, 200, 200, color=(55, 55, 255))
rectangle = shapes.Rectangle(250, 300, 400, 200, color=(255, 22, 20))
rectangle.opacity = 128
rectangle.rotation = 33
line = shapes.Line(100, 100, 100, 200, width=19)
line2 = shapes.Line(150, 150, 444, 111, width=4, color=(200, 20, 20))
star = shapes.Star(800, 400, 60, 40, num_spikes=20, color=(255, 255, 0))
widgets = circle, square, rectangle, line, line2, star@window.event
def on_draw():window.clear()for batch in widgets:batch.draw()pyglet.app.run()

接下来说说group分组参数,先看一个实例:

import pygletwindow = pyglet.window.Window(800, 500 , caption='Batch&Group')
batch = pyglet.graphics.Batch()  # 创建一个Batch对象
group = [pyglet.graphics.Group(i) for i in range(4)]   # 创建一组Group对象circle1 = pyglet.shapes.Circle(x=250, y=250, radius=80, color=(255, 0, 0, 255), batch=batch, group=group[0])
rectangle1 = pyglet.shapes.Rectangle(x=150, y=200, width=200, height=100, color=(0, 0, 255, 255), batch=batch, group=group[1])circle2 = pyglet.shapes.Circle(x=550, y=250, radius=80, color=(255, 0, 0, 255), batch=batch, group=group[3])
rectangle2 = pyglet.shapes.Rectangle(x=450, y=200, width=200, height=100, color=(0, 0, 255, 255), batch=batch, group=group[2])@window.event
def on_draw():window.clear()batch.draw()  # 在窗口上绘制batch中的所有图形对象pyglet.app.run()

运行效果:

 group参数主要用于控制控件的显示顺序,与pyglet.graphics.Group(order=i)配合使用,这里order数值越大越在上层,越小则越在下层。当然group除了排序的功能,还有包括同组绑定纹理、着色器或设置任何其他参数的功能,这些功能以后再安排学习。

到此已基本了解pyglet控件参数batch和group的用法,再来看一下graphics.Batch和graphics.Group的自带说明:

class Batch

Help on class Batch in module pyglet.graphics:

class Batch(builtins.object)
Manage a collection of drawables for batched rendering.

Many drawable pyglet objects accept an optional `Batch` argument in their
constructors. By giving a `Batch` to multiple objects, you can tell pyglet
that you expect to draw all of these objects at once, so it can optimise its
use of OpenGL. Hence, drawing a `Batch` is often much faster than drawing
each contained drawable separately.

The following example creates a batch, adds two sprites to the batch, and
then draws the entire batch::

    batch = pyglet.graphics.Batch()
    car = pyglet.sprite.Sprite(car_image, batch=batch)
    boat = pyglet.sprite.Sprite(boat_image, batch=batch)

    def on_draw():
        batch.draw()

While any drawables can be added to a `Batch`, only those with the same
draw mode, shader program, and group can be optimised together.

Internally, a `Batch` manages a set of VertexDomains along with
information about how the domains are to be drawn. To implement batching on
a custom drawable, get your vertex domains from the given batch instead of
setting them up yourself.

Methods defined here:

__init__(self)
    Create a graphics batch.

draw(self)
    Draw the batch.

draw_subset(self, vertex_lists)
    Draw only some vertex lists in the batch.

    The use of this method is highly discouraged, as it is quite
    inefficient.  Usually an application can be redesigned so that batches
    can always be drawn in their entirety, using `draw`.

    The given vertex lists must belong to this batch; behaviour is
    undefined if this condition is not met.

    :Parameters:
        `vertex_lists` : sequence of `VertexList` or `IndexedVertexList`
            Vertex lists to draw.

get_domain(self, indexed, mode, group, program, attributes)
    Get, or create, the vertex domain corresponding to the given arguments.

invalidate(self)
    Force the batch to update the draw list.

    This method can be used to force the batch to re-compute the draw list
    when the ordering of groups has changed.

    .. versionadded:: 1.2

migrate(self, vertex_list, mode, group, batch)
    Migrate a vertex list to another batch and/or group.

    `vertex_list` and `mode` together identify the vertex list to migrate.
    `group` and `batch` are new owners of the vertex list after migration.

    The results are undefined if `mode` is not correct or if `vertex_list`
    does not belong to this batch (they are not checked and will not
    necessarily throw an exception immediately).

    `batch` can remain unchanged if only a group change is desired.

    :Parameters:
        `vertex_list` : `~pyglet.graphics.vertexdomain.VertexList`
            A vertex list currently belonging to this batch.
        `mode` : int
            The current GL drawing mode of the vertex list.
        `group` : `~pyglet.graphics.Group`
            The new group to migrate to.
        `batch` : `~pyglet.graphics.Batch`
            The batch to migrate to (or the current batch).

----------------------------------------------------------------------
Data descriptors defined here:

__dict__
    dictionary for instance variables (if defined)

__weakref__
    list of weak references to the object (if defined)

class Group

Help on class Group in module pyglet.graphics:

class Group(builtins.object)
Group(order=0, parent=None)

Group of common OpenGL state.

`Group` provides extra control over how drawables are handled within a
`Batch`. When a batch draws a drawable, it ensures its group's state is set;
this can include binding textures, shaders, or setting any other parameters.
It also sorts the groups before drawing.

In the following example, the background sprite is guaranteed to be drawn
before the car and the boat::

    batch = pyglet.graphics.Batch()
    background = pyglet.graphics.Group(order=0)
    foreground = pyglet.graphics.Group(order=1)

    background = pyglet.sprite.Sprite(background_image, batch=batch, group=background)
    car = pyglet.sprite.Sprite(car_image, batch=batch, group=foreground)
    boat = pyglet.sprite.Sprite(boat_image, batch=batch, group=foreground)

    def on_draw():
        batch.draw()

:Parameters:
    `order` : int
        Set the order to render above or below other Groups.
        Lower orders are drawn first.
    `parent` : `~pyglet.graphics.Group`
        Group to contain this Group; its state will be set before this
        Group's state.

:Variables:
    `visible` : bool
        Determines whether this Group is visible in any of the Batches
        it is assigned to. If ``False``, objects in this Group will not
        be rendered.
    `batches` : list
        Read Only. A list of which Batches this Group is a part of.

Methods defined here:

__eq__(self, other)
    Return self==value.

__hash__(self)
    Return hash(self).

__init__(self, order=0, parent=None)
    Initialize self.  See help(type(self)) for accurate signature.

__lt__(self, other)
    Return self<value.

__repr__(self)
    Return repr(self).

set_state(self)
    Apply the OpenGL state change.

    The default implementation does nothing.

set_state_recursive(self)
    Set this group and its ancestry.

    Call this method if you are using a group in isolation: the
    parent groups will be called in top-down order, with this class's
    `set` being called last.

unset_state(self)
    Repeal the OpenGL state change.

    The default implementation does nothing.

unset_state_recursive(self)
    Unset this group and its ancestry.

    The inverse of `set_state_recursive`.

----------------------------------------------------------------------
Readonly properties defined here:

batches

order

----------------------------------------------------------------------
Data descriptors defined here:

__dict__
    dictionary for instance variables (if defined)

__weakref__
    list of weak references to the object (if defined)

visible

最后,使用Batch()修改一下之前两个例程:

import pygletwindow = pyglet.window.Window(800, 600, caption='绘制网络线')
color = (255, 255, 255, 255)
lines = pyglet.graphics.Batch()for y in range(0, window.height, 40):exec(f'y{y//40} = pyglet.shapes.Line(0, y, window.width, y, color=color, batch=lines)')
for x in range(0, window.width, 40):exec(f'x{x//40} = pyglet.shapes.Line(x, 0, x, window.height, color=color, batch=lines)')@window.event
def on_draw():window.clear()lines.draw()pyglet.app.run()
import pygletwindow = pyglet.window.Window(800, 600, caption='按钮模拟')GRAY = (220, 220, 220, 255)
BLACK = (0, 0, 0, 255)
RED = (255, 0, 0, 255)batch = pyglet.graphics.Batch()
pattern = pyglet.image.SolidColorImagePattern(color=GRAY)
image = pattern.create_image(100,50)
button1 = pyglet.sprite.Sprite(image, 280, 200, batch = batch)
button2 = pyglet.sprite.Sprite(image, 280+150, 200, batch = batch)
labelb1 = pyglet.text.Label('按钮1', font_size=16, x=305, y=216, color=BLACK, batch = batch)
labelb2 = pyglet.text.Label('按钮2', font_size=16, x=455, y=216, color=BLACK, batch = batch)
message = pyglet.text.Label('', font_size=16, x=20, y=16, batch = batch)@window.event
def on_draw():window.clear()batch.draw()@window.event
def on_mouse_press(x, y, button, modifier):X, Y, W, H = button1.x, button1.y, button1.width, button1.heightif x in range(X,X+W+1) and y in range(Y,Y+H+1):message.text = '你点击了按钮1'elif x in range(X+150,X+W+151) and y in range(Y,Y+H+1):message.text = '你点击了按钮2'@window.event
def on_mouse_motion(x, y, dx, dy):X, Y, W, H = button1.x, button1.y, button1.width, button1.heightif x in range(X,X+W+1) and y in range(Y,Y+H+1):labelb1.color = REDelif x in range(X+150,X+W+151) and y in range(Y,Y+H+1):labelb2.color = REDelse:labelb1.color = BLACKlabelb2.color = BLACKpyglet.app.run()

完。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/704684.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

JavaScript的math对象是什么? 有什么用

Math 对象在 JavaScript 中是一个内置的全局对象&#xff0c;它提供了多种数学相关的常量和函数。这个对象不需要我们创建&#xff0c;我们可以直接通过 Math 来访问其属性和方法。 Math 对象中包含了许多数学相关的常量&#xff0c;例如 Math.PI 代表圆周率&#xff0c;Math.…

跳跃游戏

55. 跳跃游戏 给你一个非负整数数组 nums &#xff0c;你最初位于数组的 第一个下标 。数组中的每个元素代表你在该位置可以跳跃的最大长度。 判断你是否能够到达最后一个下标&#xff0c;如果可以&#xff0c;返回 true &#xff1b;否则&#xff0c;返回 false 。 示例 1&a…

如何获取C语言中int类型的最大值

在C语言中&#xff0c;int 类型的大小通常是根据系统架构来决定的。在大多数现代系统上&#xff0c;int 通常是32位的&#xff0c;这意味着它可以表示的最大无符号整数值是 UINT_MAX&#xff0c;这个值在标准库头文件 <limits.h> 中定义。对于32位系统&#xff0c;UINT_M…

LeetCode704. 二分查找(C++)

LeetCode704. 二分查找 题目链接代码 题目链接 https://leetcode.cn/problems/binary-search/description/ 代码 class Solution { public:int search(vector<int>& nums, int target) {int left 0;int right nums.size() - 1;while(left < right){int midd…

2024.2.21- HCIA -bigdata模拟考试题

1、鲲鹏生态系统中&#xff0c;OpenEuler操作系统是华为社区开源版。 2、Elasticsearch底层基于Redis开源软件开发的。 3、ES中用于集群监控、查询、配置的插件去Head。 4、Hive支持MapReduce、Tez、Spark等执行引擎。 5、Hbase的特点是一个分布式&#xff0c;基于列示存储…

外包工作两个月,技术退步让我决心改变

大家好&#xff0c;我是一名大专生&#xff0c;2019年通过校招进入了湖南的一家软件公司。在这里&#xff0c;我从事了接近4年的功能测试工作。然而&#xff0c;今年8月份&#xff0c;我深刻地意识到&#xff0c;我不能继续这样下去了。 长时间在一个舒适的环境里&#xff0c;…

数据库系统概论(超详解!!!) 第一节 绪论

1.四个基本概念 1.数据&#xff08;Data&#xff09; 数据&#xff08;Data&#xff09;是数据库中存储的基本对象 数据的定义&#xff1a;描述事物的符号记录 数据的种类&#xff1a;数字、文字、图形、图像、音频、视频、学生的档案记录等 数据的含义称为数据的语义&…

如何在Tomcat中配置和使用Session共享!

如何在Tomcat中配置和使用Session共享&#xff01; Tomcat的Session共享是指多个不同的Web应用程序能够访问同一个用户会话&#xff08;User Session&#xff09;中的数据。这在分布式系统或者需要跨应用共享用户状态的场景中非常有用。以下是一篇关于如何在Tomcat中配置和使用…

网络编程中的常用的头文件

#include <arpa/inet.h> 是一个 C 语言标准库头文件。 位于 POSIX 标准库中&#xff0c;用于网络编程。它包含了一些函数原型和定义&#xff0c;用于处理 IP 地址的转换和操作。 <arpa/inet.h> 头文件提供了以下函数&#xff1a; IP 地址转换函数&#xff1a;in…

电机控制-----电机极对数,相电感,相电阻以及磁链常数的测量

电机控制-----电机极对数&#xff0c;相电感&#xff0c;相电阻以及磁链常数的测量 我们在做电机控制的时候&#xff0c;拿到一个电机首先要知道它的参数&#xff0c;然后才能进行相应的开发&#xff0c;我这里介绍的是通过平常常用的手段去获得电机的参数&#xff1a;极对数&…

记生产OOM的故障分析

一、引言 生产上告警&#xff0c;交易堵塞&#xff0c;服务无响应&#xff0c;使用jstack、jmap、jhat命令进行故障分析。 Java虚拟机&#xff08;Java Virtual Machine&#xff0c;简称JVM&#xff09;作为Java语言的核心组件&#xff0c;为Java程序提供了运行环境和内存管理…

docker存储驱动

目录 一、写时复制和用时分配 二、联合文件系统 2.1、aufs ​编辑 2.2、分层的问题 2.3、overlay 2.4 文件系统区别 三、容器跑httpd案例 3.1、案例1&#xff1a;端口映射 3.2、案例2&#xff1a;制作httpd应用镜像 3.3、案例3&#xff1a;docker数据卷挂载 3.4、案…

【hot100】跟着小王一起刷leetcode -- 49. 字母异位词分组

【【hot100】跟着小王一起刷leetcode -- 49. 字母异位词分组 49. 字母异位词分组题目解读解题思路代码实现 总结 49. 字母异位词分组 题目解读 49. 字母异位词分组 ok&#xff0c;兄弟们&#xff0c;咱们来看看这道题&#xff0c;很明显哈&#xff0c;这里的关键词是字母异位…

后缀表达式

一、题目描述 P8683 [蓝桥杯 2019 省 B] 后缀表达式 二、算法简析 显然&#xff0c;这道题要用贪心思想。想当然的&#xff0c;我们会先进行降序排序&#xff0c;将大的相加&#xff0c;在减去小的。然而&#xff0c;这种想法是错误的。因为这道题要求的是后缀表达式的最大值…

PYTHON 120道题目详解(94-96)

94.Python中的递归函数是什么&#xff1f;请给出一个例子。 递归函数是一种特殊的函数&#xff0c;它在其定义中会调用自身。这种调用通常是在函数的某个条件语句中&#xff0c;当满足某个条件时&#xff0c;函数会调用自身来处理更小或更简单的子问题。递归函数通常用于解决可…

docker 安装gitlab

Docker方式安装参考&#xff1a; https://docs.gitlab.com/ee/install/docker.html https://docs.gitlab.cn/jh/install/docker.html 1.确保docker与docker-compose安装运行正常 # docker version # docker-compose version 2.关闭防火墙 # sudo systemctl stop firewalld …

分布式任务调度的几种实现(Redis实现分布式锁 MySQL实现任务调度 负载均衡)

需求背景 现在有一个计算搜索词热榜的任务&#xff0c;该服务部署在了多个节上&#xff0c;希望只有一个节点在执行这个任务。 常见方案 使用Redis实现分布式锁方案 使用一个分布式锁&#xff0c;确保整个分布式环境下&#xff0c;只有一个节点能够拿到锁。节点先抢占分布式…

《最新出炉》系列初窥篇-Python+Playwright自动化测试-27-处理单选和多选按钮-番外篇

1.简介 前边几篇文章是宏哥自己在本地弄了一个单选和多选的demo&#xff0c;然后又找了网上相关联的例子给小伙伴或童鞋们演示了一下如何使用playwright来处理单选按钮和多选按钮进行自动化测试&#xff0c;想必大家都已经掌握的八九不离十了吧。这一篇其实也很简单&#xff1a…

结构体(一)

结构体的声明&#xff1a; 结构体由一系列成员组成&#xff0c;每个成员的类型可以是基本数据类型&#xff08;int&#xff0c;char&#xff0c;double…&#xff09;或复合数据类型&#xff08;结构体&#xff0c;枚举&#xff0c;数组…&#xff09;。比如如下例子&#xf…

numpy快速使用

numpy是一个开源的Python科学计算库&#xff0c;它提供了高效的多维数组对象&#xff0c;以及计算、操作数组数据的工具。以下是一些numpy的常见用法示例&#xff1a; 导入numpy库&#xff1a; import numpy as np创建数组&#xff1a; arr np.array([1, 2, 3, 4, 5]) # 一…