通过在Pygame中绘制线条来填充三角形

如何解决通过在Pygame中绘制线条来填充三角形

我的最终目标是填充构成3d网格的三角形。为了弄清楚如何做到这一点,我决定生成一个随机三角形并将其填充。稍后,在一些文章和youtube视频之后,我认为通常是通过计算三角形不同线之间的x和y并填充它们来完成的。像素。但是在我的情况下,pygame已经具有画线功能,因此我可以简单地计算出构成最长边的两个点之间的所有点,并从这些点到第三点画线,或者我想。这是复制和粘贴代码:

import pygame
from random import randint
from math import sqrt
pygame.init()

D = pygame.display.set_mode((1200,600))

class Pos:
    def __init__(self,x,y):
        self.x = x
        self.y = y

def genTriangle():
    point1 = Pos(randint(50,1000),randint(50,550))
    point2 = Pos(randint(50,550))
    point3 = Pos(randint(50,550))
    return [point1,point2,point3]

tri = genTriangle()

def getLength(p1,p2):
    x = p1.x - p2.x
    y = p1.y - p2.y
    return sqrt(x**2 + y**2)
    
def fill(tri):
    len01 = getLength(tri[0],tri[1]) #Length between tri[0] and tri[1]
    len12 = getLength(tri[1],tri[2]) #Length between tri[1] and tri[2]
    len20 = getLength(tri[2],tri[0]) #Length between tri[2] and tri[0]

    # Assinging the points making up the longest side to p1 and p3 
    # and the third point to p2 becasue that is how the calculation is carried
    # out later (i feel like this is now the best way to figure out the longest side
    # so any help with this would be appreciated as well)
    if len01 > len12 and len20:
        p1 = tri[0]
        p2 = tri[2]
        p3 = tri[1]

    elif len12 > len20 and len01:
        p1 = tri[1]
        p2 = tri[0]
        p3 = tri[2]

    elif len20 > len01 and len12:
        p1 = tri[2]
        p2 = tri[1]
        p3 = tri[0]

    # calculates all the points along the longest side of the triangle
    # and draws lines from those points to  the third point of the triangle
    for x in range(p1.x,p3.x+1):
        m = ((p1.y - p3.y)/ (p1.x - p3.x)) # slope
        y = m*(x - p3.x) + p3.y # rearranged point-slope formula 
        pygame.draw.line(D,(255,0),(int(x),int(y)),(p2.x,p2.y),1) 


while True:
    pygame.event.get()
    D.fill((255,255,255))
    fill(tri)
    points = [(point.x,point.y) for point in tri]
    pygame.draw.lines(D,(0,True,points,1)
    pygame.display.flip()

结果是,有时可以正确地填充三角形而没有任何问题。有时三角形是填充的,但也有一些未填充的像素,有时三角形根本不填充。感谢您的帮助。

解决方法

一个明显的错误是线

for x in range(p1.x,p3.x+1):

如果您未在函数range中指定 step 参数,则 start 必须小于 stop 。计算最小和最大坐标:

for x in range(min(p1.x,p3.x),max(p1.x,p3.x)+1):
    m = ((p1.y - p3.y)/ (p1.x - p3.x)) # slope
    y = m*(x - p3.x) + p3.y # rearranged point-slope formula 
    pygame.draw.line(D,(255,0),(int(x),int(y)),(p2.x,p2.y),1) 

此外,条件if len01 > len12 and len20:不能满足您的期望。它必须是if len01 > len12 and len01 > len20:
参见How to test multiple variables against a value?


您必须评估是否为abs(p3.x-p1.x) > abs(p3.y-p1.y)。如果条件为True,则沿x轴进行迭代,否则沿y轴进行迭代:

def fill(tri):
    len01 = getLength(tri[0],tri[1]) #Length between tri[0] and tri[1]
    len12 = getLength(tri[1],tri[2]) #Length between tri[1] and tri[2]
    len20 = getLength(tri[2],tri[0]) #Length between tri[2] and tri[0]

    # Assinging the points making up the longest side to p1 and p3 
    # and the third point to p2 becasue that is how the calculation is carried
    # out later (i feel like this is now the best way to figure out the longest side
    # so any help with this would be appreciated as well)
    if len01 > len12 and len01 > len20:
        p1,p2,p3 = tri[0],tri[2],tri[1]
    elif len12 > len20 and len12 > len01:
        p1,p3 = tri[1],tri[0],tri[2]
    elif len20 > len01 and len20 > len12:
        p1,p3 = tri[2],tri[1],tri[0]

    # calculates all the points along the longest side of the triangle
    # and draws lines from those points to  the third point of the triangle
    if abs(p3.x-p1.x) > abs(p3.y-p1.y):
        for x in range(min(p1.x,p3.x)+1):
            m = ((p1.y - p3.y)/ (p1.x - p3.x)) # slope
            y = m*(x - p3.x) + p3.y # rearranged point-slope formula 
            pygame.draw.line(D,1) 
    else:
        for y in range(min(p1.y,p3.y),max(p1.y,p3.y)+1):
            m = ((p1.x - p3.x)/ (p1.y - p3.y)) # slope
            x = m*(y - p3.y) + p3.x # rearranged point-slope formula 
            pygame.draw.line(D,1)

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


依赖报错 idea导入项目后依赖报错,解决方案:https://blog.csdn.net/weixin_42420249/article/details/81191861 依赖版本报错:更换其他版本 无法下载依赖可参考:https://blog.csdn.net/weixin_42628809/a
错误1:代码生成器依赖和mybatis依赖冲突 启动项目时报错如下 2021-12-03 13:33:33.927 ERROR 7228 [ main] o.s.b.d.LoggingFailureAnalysisReporter : *************************** APPL
错误1:gradle项目控制台输出为乱码 # 解决方案:https://blog.csdn.net/weixin_43501566/article/details/112482302 # 在gradle-wrapper.properties 添加以下内容 org.gradle.jvmargs=-Df
错误还原:在查询的过程中,传入的workType为0时,该条件不起作用 <select id="xxx"> SELECT di.id, di.name, di.work_type, di.updated... <where> <if test=&qu
报错如下,gcc版本太低 ^ server.c:5346:31: 错误:‘struct redisServer’没有名为‘server_cpulist’的成员 redisSetCpuAffinity(server.server_cpulist); ^ server.c: 在函数‘hasActiveC
解决方案1 1、改项目中.idea/workspace.xml配置文件,增加dynamic.classpath参数 2、搜索PropertiesComponent,添加如下 <property name="dynamic.classpath" value="tru
删除根组件app.vue中的默认代码后报错:Module Error (from ./node_modules/eslint-loader/index.js): 解决方案:关闭ESlint代码检测,在项目根目录创建vue.config.js,在文件中添加 module.exports = { lin
查看spark默认的python版本 [root@master day27]# pyspark /home/software/spark-2.3.4-bin-hadoop2.7/conf/spark-env.sh: line 2: /usr/local/hadoop/bin/hadoop: No s
使用本地python环境可以成功执行 import pandas as pd import matplotlib.pyplot as plt # 设置字体 plt.rcParams['font.sans-serif'] = ['SimHei'] # 能正确显示负号 p
错误1:Request method ‘DELETE‘ not supported 错误还原:controller层有一个接口,访问该接口时报错:Request method ‘DELETE‘ not supported 错误原因:没有接收到前端传入的参数,修改为如下 参考 错误2:cannot r
错误1:启动docker镜像时报错:Error response from daemon: driver failed programming external connectivity on endpoint quirky_allen 解决方法:重启docker -> systemctl r
错误1:private field ‘xxx‘ is never assigned 按Altʾnter快捷键,选择第2项 参考:https://blog.csdn.net/shi_hong_fei_hei/article/details/88814070 错误2:启动时报错,不能找到主启动类 #
报错如下,通过源不能下载,最后警告pip需升级版本 Requirement already satisfied: pip in c:\users\ychen\appdata\local\programs\python\python310\lib\site-packages (22.0.4) Coll
错误1:maven打包报错 错误还原:使用maven打包项目时报错如下 [ERROR] Failed to execute goal org.apache.maven.plugins:maven-resources-plugin:3.2.0:resources (default-resources)
错误1:服务调用时报错 服务消费者模块assess通过openFeign调用服务提供者模块hires 如下为服务提供者模块hires的控制层接口 @RestController @RequestMapping("/hires") public class FeignControl
错误1:运行项目后报如下错误 解决方案 报错2:Failed to execute goal org.apache.maven.plugins:maven-compiler-plugin:3.8.1:compile (default-compile) on project sb 解决方案:在pom.
参考 错误原因 过滤器或拦截器在生效时,redisTemplate还没有注入 解决方案:在注入容器时就生效 @Component //项目运行时就注入Spring容器 public class RedisBean { @Resource private RedisTemplate<String
使用vite构建项目报错 C:\Users\ychen\work>npm init @vitejs/app @vitejs/create-app is deprecated, use npm init vite instead C:\Users\ychen\AppData\Local\npm-