python - 在 pygame 中移动 cometd-LMLPHP

我想要一个 pygame 中的 cometd 在屏幕上射击。这是我的 cometd 类

class Commet:
    def __init__(self):
        self.x  = -10
        self.y = 10
        self.radius = 20
        self.commet = pygame.image.load(r"C:\Users\me\OneDrive\Documents\A level python codes\final game\commet.png")
        self.commet = pygame.transform.scale(self.commet, (self.radius, self.radius))
        self.drop = 0.0000009
        self.speed = 2
        self.pos = 0
        self.commets = []

然后我在 self.commets 列表中添加了 20 个 commets。
  def tail(self, n): # n is a variable used to denote the length of the self.commets
        for i in range(n):
            if len(self.commets) <= n - 1:
                self.commets.append(Commet())

我有两个问题。第一个问题是移动 cometd 。为了移动它,我做了这个
def move_tail(self):
    for c in  self.commets:
        c.x += self.speed
    for i in range(len(self.commets) - 1):
         self.commets[i].y += ((self.commets[i + 1].x) ** 2) * self.drop

对于 x 坐标,我只是在每帧的值上加 2。但是,对于 cometd 的 y 值,我希望它产生类似尾部的跟随效果。我尝试将 cometd 的 y 值分配给我们在列表 x 中所指的 cometd 上方的索引位置中 cometd 的 self.commets 值的平方。我希望 cometd 沿着一般的 x = y * 相互跟随*2 二次曲线。它们确实遵循曲线,但都以相同的速率(我预计它们会以不同的速率跟随,因为所有 cometd 都有不同的 x 值),这给我带来了类似尾部的效果。我怎样才能产生这种尾部状的效果?

我的问题的第二部分是我希望第一个之后的 cometd 变得越来越小。我尝试减小 radius 值,该值用于缩放我导入的图像。代码看起来像这样
    # Decrease radius
    for i in range(n):
        self.commets[i].radius = i + 1

当我在控制台上打印出 cometd 半径的值时,它们的范围从 1 到 20,正如我所期望的那样,但是屏幕上显示的图像大小对于列表中的所有 cometd 都是相同的。以下代码是我如何对 cometd 进行 blit
  for i in range(n):
        self.commets[i].pos = i * 10 #  This line maintains a certain x- distance between commets

    for c in self.tails:
        D.blit(c.commet, (c.x - c.pos, c.y))

        if self.pos >= n:
            self.pos = n

最佳答案

假设您希望 cometd 在全高清屏幕上从左到右飞行。

cometd 应从左侧 y 坐标 900 处开始,然后在 x=1400 和 y = 100 处达到最高点,然后在屏幕右侧下降到 600。

python - 在 pygame 中移动 cometd-LMLPHP

抛物线通常是 y = ax²+bx+c。

为了独立于屏幕分辨率,您当然会从某个百分比计算这些值,例如 900 ~ 屏幕高度 * 83%、600 ~ 屏幕高度 * 55%、1400 ~ 屏幕宽度 * 73%、100 ~ 屏幕高度 * 9 %

给定三个点,您可以计算抛物线:

class ParabolaFrom3Points:
    def __init__(self, points: list):
        self.a = (points[0][0] * (points[1][1] - points[2][1]) + points[1][0] * (
                points[2][1] - points[0][1]) + points[2][0] * (points[0][1] - points[1][1])) / (
                         (points[0][0] - points[1][0]) * (points[0][0] - points[2][0]) * (
                         points[2][0] - points[1][0]))
        self.b = (points[0][0] ** 2 * (points[1][1] - points[2][1]) + points[1][0] ** 2 * (
                points[2][1] - points[0][1]) + points[2][0] ** 2 * (points[0][1] - points[1][1])) / (
                         (points[0][0] - points[1][0]) * (points[0][0] - points[2][0]) * (
                         points[1][0] - points[2][0]))
        self.c = (points[0][0] ** 2 * (points[1][0] * points[2][1] - points[2][0] * points[1][1]) +
                  points[0][0] * (points[2][0] ** 2 * points[1][1] - points[1][0] ** 2 * points[2][1]) +
                  points[1][0] * points[2][0] * points[0][1] * (points[1][0] - points[2][0])) / (
                         (points[0][0] - points[1][0]) * (points[0][0] - points[2][0]) * (
                         points[1][0] - points[2][0]))

    def y(self, x: int) -> int:
        return int(self.a * x ** 2 + self.b * x + self.c)

那么 cometd 就很简单了。它只需要知道它的抛物线函数,然后就可以从 x 计算 y。
class Comet:
    def __init__(self, radius: int, para: ParabolaFrom3Points):
        self.x = -radius  # Be invisible at the beginning
        self.radius = radius
        self.para = para

    def move(self, x):
        self.x = x

    def paint(self, screen):
        x = self.x
        radius = self.radius
        for tail in range(20):
            pygame.draw.circle(screen, [255, 255, 255], (int(x), self.para.y(x)), radius)
            x = x - radius / 2
            radius -= 1

测试代码:
import pygame

pygame.init()
pygame.fastevent.init()
clock = pygame.time.Clock()
window = pygame.display.set_mode((1920, 1080))

pygame.display.set_caption('Comet example')
comet = Comet(20, ParabolaFrom3Points([(0, 1080 * 0.83), (1920 * 0.73, 1080 * 0.12), (1920, 1080 * 0.55)]))
for x in range(-20, 1920 + 200, 3):
    comet.move(x)
    comet.paint(window)
    clock.tick(90)
    pygame.display.flip()
    window.fill([0, 0, 0])

pygame.quit()

关于python - 在 pygame 中移动 cometd ,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/60141474/

10-11 08:48