Pygame 教程(4)拓展:使用 subsurface 方法

2023-11-19

原文:图像传输和绘制文本

在本拓展文章,将展示如何使用pygame.Surface.subsurface方法实现原文中的更改线条粗细步骤。

pygame.Surface.subsurface将获取原Surface对象的子Surface它与原Surface共享像素,原图像的操作也将影响到改子图像。如果不想该子图像与原图像有任何的关联,可以在调用此方法后使用copy方法创建副本。

原文实例中,可以通过subsurface方法截取用户绘制的部分,然后重绘屏幕及色板,最后再将截取的部分重新绘制到屏幕上。

draw_text方法保留原来的样子,并添加change_line_width方法,代码如下:

# ...

def draw_text(self):
    surf = self.font.render(f'Line width: {self.width}', True, (0, 0, 0))
    rect = surf.get_rect()
    rect.right = PALETTE_RECT.right - GAP
    rect.centery = PALETTE_RECT.centery
    fill_rect = pygame.Rect(rect.left - 5, rect.top - 5, rect.width + 10, rect.height + 10)
    pygame.draw.rect(self.screen, (255, 255, 255), fill_rect)
    self.screen.blit(surf, rect)
    
def change_line_width(self):
    board_rect = (0, PALETTE_RECT.bottom, 800, 800 - PALETTE_HEIGHT)
    board = self.screen.subsurface(board_rect).copy()
    self.screen.fill((255, 255, 255))
    self.circles.clear()
    self.draw_colors()
    self.draw_text()
    self.screen.blit(board, board_rect)

# ...

修改on_key_down方法,将调用draw_text改为调用change_line_width

def on_key_down(self, key):
    if self.drawing:
        return
    if key == pygame.K_UP:
        self.width += 1
        self.width = clamp(self.width, 1, LINE_WIDTH_MAX)
        self.change_line_width()
    elif key == pygame.K_DOWN:
        self.width -= 1
        self.width = clamp(self.width, 1, LINE_WIDTH_MAX)
        self.change_line_width()

完整代码如下:

import sys

import pygame

COLORS = (
    (0, 0, 0),
    (255, 0, 0),
    (0, 255, 0),
    (0, 0, 255),
    (255, 255, 0),
    (255, 0, 255),
    (0, 255, 255),
    (255, 255, 255)
)  # 色板中的所有颜色

RADIUS = 15  # 色板中设置颜色的圆圈的半径
GAP = 15  # 色板中相邻的圆圈之间的空隙
CENTER_DIST = RADIUS * 2 + GAP  # 色板中相邻的圆圈的中心之间的距离
PALETTE_RECT = pygame.Rect(0, 0, 800, RADIUS * 2 + GAP * 2)  # 色板的矩形坐标
PALETTE_HEIGHT = PALETTE_RECT.height  # 色板的矩形的高度
LINE_WIDTH_MAX = 20  # 线条的最大粗度


def clamp(value, low, high):
    return low if value < low else (high if value > high else value)


def clamp_pos(pos):
    y = clamp(pos[1], PALETTE_HEIGHT, 800)
    return (pos[0], y)


def dist_between_points(point1, point2):
    return ((point1[0] - point2[0]) ** 2 + (point1[1] - point2[1]) ** 2) ** 0.5


class DrawingBoard:
    def __init__(self):
        pygame.init()
        self.screen = pygame.display.set_mode((800, 800))
        self.screen.fill((255, 255, 255))
        pygame.display.set_caption('Drawing Board')

        self.color = (0, 0, 0)  # 画笔的颜色
        self.width = 1  # 线条的粗细
        self.last_pos = self.pos = (0, 0)  # 鼠标上一帧的位置以及当前位置
        self.circles = []  # 色板中由圆圈的中心和对应颜色的元组组合而成。
        self.font = pygame.font.SysFont('Times New Roman', 20)  # 字体
        self.drawing = False  # 用户是否正在绘画

        self.draw_colors()
        self.draw_text()

    def draw_colors(self):
        for i in range(len(COLORS)):
            center = (i * CENTER_DIST + GAP + RADIUS, GAP + RADIUS)
            pygame.draw.circle(self.screen, COLORS[i], center, RADIUS)
            self.circles.append((center, COLORS[i]))

        pygame.draw.line(self.screen, (0, 0, 0), (0, PALETTE_HEIGHT), (800, PALETTE_HEIGHT))

    def draw_text(self):
        surf = self.font.render(f'Line width: {self.width}', True, (0, 0, 0))
        rect = surf.get_rect()
        rect.right = PALETTE_RECT.right - GAP
        rect.centery = PALETTE_RECT.centery
        fill_rect = pygame.Rect(rect.left - 5, rect.top - 5, rect.width + 10, rect.height + 10)
        pygame.draw.rect(self.screen, (255, 255, 255), fill_rect)
        self.screen.blit(surf, rect)
        
    def change_line_width(self):
        board_rect = (0, PALETTE_RECT.bottom, 800, 800 - PALETTE_HEIGHT)
        board = self.screen.subsurface(board_rect).copy()
        self.screen.fill((255, 255, 255))
        self.circles.clear()
        self.draw_colors()
        self.draw_text()
        self.screen.blit(board, board_rect)

    def on_mouse_down(self, pos):
        if PALETTE_RECT.collidepoint(pos):
            for center, color in self.circles:
                if dist_between_points(pos, center) < RADIUS:
                    self.color = color
        else:
            self.last_pos = self.pos = pos
            self.drawing = True
            
    def on_mouse_move(self, pos):
        if self.drawing:
            self.pos = clamp_pos(pos)
            pygame.draw.line(self.screen, self.color, self.last_pos, self.pos, self.width)
            self.last_pos = clamp_pos(pos)
            
    def on_key_down(self, key):
        if self.drawing:
            return
        if key == pygame.K_UP:
            self.width += 1
            self.width = clamp(self.width, 1, LINE_WIDTH_MAX)
            self.change_line_width()
        elif key == pygame.K_DOWN:
            self.width -= 1
            self.width = clamp(self.width, 1, LINE_WIDTH_MAX)
            self.change_line_width()

    def process_events(self):
        for event in pygame.event.get():
            if event.type == pygame.QUIT:
                pygame.quit()
                sys.exit()
            elif event.type == pygame.MOUSEBUTTONDOWN:
                self.on_mouse_down(event.pos)
            elif event.type == pygame.MOUSEMOTION:
                self.on_mouse_move(event.pos)
            elif event.type == pygame.MOUSEBUTTONUP:
                self.drawing = False
            elif event.type == pygame.KEYDOWN:
                self.on_key_down(event.key)

    def run(self):
        while True:
            self.process_events()
            pygame.display.update()


if __name__ == '__main__':
    app = DrawingBoard()
    app.run()

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

Pygame 教程(4)拓展:使用 subsurface 方法 的相关文章

随机推荐

  • 对于单向链表的排序与去重

    include bits stdc h using namespace std struct node int num node next class Chain public Chain head tail new node void G
  • 【计算机科学】【2017】变分推理与深度学习

    本文为荷兰阿姆斯特丹大学 作者 Kingma D P 的博士论文 共174页 在本文中 我们提出了变分 贝叶斯 推理 生成建模 表示学习 半监督学习和随机优化问题的新的解决方案 我们提出了一种有效的变分推理算法 Kingmaand Well
  • 解析目标检测全流程!附代码数据

    关注后 星标 Datawhale 每日干货 每月组队学习 不错过 Datawhale干货 作者 王程伟 算法工程师 Datawhale成员 在计算机视觉中 红外弱小目标检测是一个重要的方向 但直到近一两年 才开始运用一些深度学习的方法 深度
  • 二叉树的递归遍历、非递归遍历、层次遍历

    1 递归遍历 2 非递归遍历 3 层次遍历 1 递归遍历 在使用递归遍历的时候 每个节点会经过三次 public class PreInPosTraversal public static class Node public int val
  • Java 基础--- Object Class

    Java 基础 Object Class Object Class 结构 toString clone finalize getClass equals Object obj hashCode 什么是hashcode hashcode的主要
  • 【MACOS(M1)编译Risc-v版OpenOCD】

    MACOS编译Risc v版OpenOCD 准备 1 执行顺序 常见问题 问题1 AC PROG CC C99 告警 2 问题2 texinfo 版本不匹配 问题2 libtool版本不匹配 问题3 编译错误 验证一下 准备1 Instal
  • document this 插件 mac_又一个折腰的 VSCode 插件

    1 最近这两周周末都在折腾 VSCode 白板插件 其主要原理是将 Excalidraw 1 嵌入到 VSCode 的 WebView 中 如果所有功能正常 那么相比于在 Web 中使用 在 VSCode 中将会更方便 截至目前 该插件的功
  • Power BI——SUMX函数(对列操作)

    1 语法 说明 第一个参数为被运算的表 table 第二个参数是对表中的每一行计算的表达式 2 步骤 SUMX 将迭代第一个参数中指定的表 一次一行 并完成第二个参数中指定的计算 3 案例 总销售额SUMX SUMX 销售表 销售表 数量
  • 【路径规划】基于A*算法和Dijkstra算法的路径规划(Python代码实现)

    个人主页 研学社的博客 欢迎来到本博客 博主优势 博客内容尽量做到思维缜密 逻辑清晰 为了方便读者 座右铭 行百里者 半于九十 本文目录如下 目录 1 概述 2 运行结果 3 参考文献 4 Python代码实现 1 概述 Dijkstr
  • 基于 tensorflow 的鲜花识别

    本文中 采用了卷积神经网络的深度学习方法来实现对鲜花种类的高精度识别 先是对原始图像进行预处理 然后是以LeNet 5 卷积神经网络模型为基础建立网络模型 再进行模型训练后得到鲜花识别分类器 最后分析实验结果并与其它花卉识别的算法进行对比
  • MySQL的JDBC操作入门案例 IDEA 实现

    package cn itcast jdbc import java sql public class JdbcDemo01 public static void main String args throws SQLException 1
  • 设计模式学习笔记(一)如何评判代码质量的好坏?

    1 评判代码质量好坏的角度 在目前的面板项目中 经常听到同事用 代码写的很烂 这是什么鬼 我去 这个思路很棒啊 等字眼来评判项目中部分代码逻辑的好坏 经过阅读一些文章和学习技术大神分享的内容 了解到真正评判代码好坏不应该笼统的用 好 烂 这
  • java实现starteam配置库中checkout对应的文件

    1 所属jar包 starteam sdk 5 2 jar 2 样例代码 package com comtop qcms base starteam import java io IOException import com comtop
  • 结果解读_染色体CNV检测结果解读

    染色体CNV检测的结果 大致可以分三类 完全正常或多态性 致病性片段 及 临床意义未明 值得一提的是 由于分子遗传学检测技术的发展 比如染色体CNV检测 染色体层面的产前诊断的时间不再局限于细胞培养的最佳时间 18 26周 脐血穿刺也几乎被
  • 编译mysql出现CMake Error at cmake/readline.cmake:83 (MESSAGE)

    本文转载至 http blog 163 com sz2273 pr blog static 41264296201361354426670 Could NOT find Curses missing CURSES LIBRARY CURSE
  • 【业务功能篇104】 补充【业务功能篇99】微服务-springcloud-springboot-电商订单模块--整合支付

    在前面我们业务功能篇98 99中 我们介绍了电商项目中的订单模块服务 那么最后就是需要进行支付动作 那么我们这里就通过订阅第三方平台支付宝的支付调用接口功能 来进一步完成订单提交后的支付动作 支付宝的接口使用可以登录官网开发指南详情去了解
  • java lazy_Spring注解之@Lazy注解使用解析

    Lazy用于指定该Bean是否取消预初始化 主要用于修饰Spring Bean类 用于指定该Bean的预初始化行为 使用该Annotation时可以指定一个boolean型的value属性 该属性决定是否要预初始化该Bean lazy代表延
  • 如何用3个方法做到照片数据恢复…

    照片数据恢复是否能够做到 现在电脑已经成为社会的主流工具 因为很多工作有已经通过互联网的桥梁达到了更高的利益 基本现在的生活已经离不开电脑的存在 当我们电脑中的照片或者相机什么的照片删除之后该怎么办 本期要给大家介绍一下照片丢失该怎么办 不
  • NHCP H4: Network Resource Management Topic

    NHCP H4 Network Resource Management Topic Cloud computing Single Choice T F items Single Choice 1 部署华为云计算解决方案时 一般将服务器BMC
  • Pygame 教程(4)拓展:使用 subsurface 方法

    原文 图像传输和绘制文本 在本拓展文章 将展示如何使用pygame Surface subsurface方法实现原文中的更改线条粗细步骤 pygame Surface subsurface将获取原Surface对象的子Surface它与原S