Python / Pygame: l oop поверх копии списка Sprite - PullRequest
1 голос
/ 01 апреля 2020

Я пытаюсь создать поток дождевых капель, чтобы дать ощущение дождя, используя код ниже. Дождевая капля используется для моделирования капли дождя. У него есть метод update () для увеличения координаты y, чтобы при вызове этого метода он выглядел так, как будто он падает на экран.

from pygame.sprite import Sprite
class Raindrop(Sprite):
    """A class to represent a single raindrop."""

    def __init__(self, rain_game):
        """Initialize the raindrop and set its starting position."""
        super().__init__()
        self.screen = rain_game.screen
        self.settings = rain_game.settings

        # Load the raindrop image and set its rect attribute
        self.image = pygame.image.load('raindrop.bmp')
        self.rect = self.image.get_rect()

        #Start each raindrop near the top left of the screen.
        self.rect.x = self.rect.width
        self.rect.y = self.rect.height

        #Store the raindrop`s exact horizontal and vertical positions.
        self.x = self.rect.x
        self.y = self.rect.y

    def update(self):
        """Shift raindrops to one level below"""
        self.y += self.settings.rain_speed
        self.rect.y = self.y

class Settings:
    """A Class to store all settings for a Sky Full of Stars."""

    def __init__(self):
        """Initialize the game`s settings."""
        # Screen setiings
        self.screen_width = 1200
        self.screen_height = 800
        self.bg_color = (255, 255, 255)
        self.rain_speed = 1.0

У меня проблема с методом remove_add_raindrops, приведенным ниже код. Он работает, как и ожидалось, но только для 4 рядов капель дождя в моем ноутбуке. После 4 строк этот код перестает печатать капли дождя на экране. Поэтому я вижу только 8 рядов капель дождя, падающих сверху вниз на экран.

import sys

import pygame

from settings import Settings
from raindrop import Raindrop

class Raining():
    """Overall class to display a sky with raindrops."""

    def __init__(self):
        """Initialize the game, and create game resources."""
        pygame.init()
        self.settings = Settings()

        self.screen = pygame.display.set_mode((0, 0), pygame.FULLSCREEN)
        self.settings.screen_width = self.screen.get_rect().width
        self.settings.screen_height = self.screen.get_rect().height
        pygame.display.set_caption("It is raining today")

        self.raindrops = pygame.sprite.Group()

        self._create_raindrops()

    def _create_raindrop(self, raindrop_number, row_number):
        """Create a raindrop and place it in the row."""
        raindrop = Raindrop(self)
        raindrop_width, raindrop_height = raindrop.rect.size
        raindrop.x = raindrop_width + 2 * raindrop_width * raindrop_number
        raindrop.rect.x = raindrop.x
        raindrop.y = raindrop_height + 2 * raindrop_height * row_number
        raindrop.rect.y = raindrop.y
        self.raindrops.add(raindrop)

    def _create_raindrops(self):
        """Create a full screen fo raindrops."""
        # Create a raindrop and find the number of raindrops in a row.
        # Spacing between each raindrop is equal to one raindrop width.
        raindrop = Raindrop(self)
        raindrop_width, raindrop_height = raindrop.rect.size
        available_space_x = self.settings.screen_width - (2 * raindrop_width)
        number_raindrops_x = available_space_x // (2 * raindrop_width)

        # Determine the number of rows of raindrops that fit on the screen.
        available_space_y = (self.settings.screen_height - 2 * raindrop_height)
        number_rows = available_space_y // (2 * raindrop_height)

        # Create a full screen of raindrops.
        for row_number in range(number_rows):
            for raindrop_number in range (number_raindrops_x):
                self._create_raindrop(raindrop_number, row_number)


    def _remove_add_raindrops(self):
        """
        Get rid of old raindrops. 
        Then update their y coordinate and append to the raindrops sprite group
        """ 
        for raindrop in self.raindrops.copy():
            if raindrop.rect.y >= self.settings.screen_height:
                self.raindrops.remove(raindrop)
                raindrop.rect.y -= self.settings.screen_height
                self.raindrops.add(raindrop)

    def run_game(self):
        """Start the main loop for the game."""
        while True:
            self._check_events()
            self._update_raindrops()
            self._remove_add_raindrops()
            self._update_screen()
            print(f"Nr of remaining raindrops: {len(self.raindrops)}")
            #Make the most recently drawn screen visible.
            pygame.display.flip()

    def _check_events(self):
        """Respond to keypresses and mouse events."""
        for event in pygame.event.get():
            if event.type == pygame.QUIT:
                sys-exit()
            elif event.type == pygame.KEYDOWN:
                self._check_keydown_events(event)

    def _check_keydown_events(self, event):
        """Respond to keypresses."""
        if event.key == pygame.K_q:
            sys.exit()

    def _update_raindrops(self):
        """Update the positions of all raindrops."""
        self.raindrops.update()

    def _update_screen(self):
        """Update images on the screen, and flip to the new screen."""
        self.screen.fill(self.settings.bg_color)
        self.raindrops.draw(self.screen)

if __name__ == '__main__':
    # Make a game instance, and run the game.
    rain_game = Raining()
    rain_game.run_game()

Я также напечатал количество капель дождя в списке, и оно всегда равно 40. Я полагаю, это означает, что я продолжаю добавлять новые капли дождя с новой координатой y после удаления тех, которые скользят под экраном.

Любые подсказки?

Обновление:

Я удалил метод _remove_add_raindrops и переписал метод обновления в классе дождевой капли, как показано ниже:

def update(self):
    """Shift raindrops to one level below"""
    self.y += self.settings.rain_speed
    self.rect.y = self.y
    if self.rect.top >= self.screen.get_rect().bottom:
        self.rect.y -= self.settings.screen_height

Это все то же поведение.

Ответы [ 2 ]

1 голос
/ 01 апреля 2020

_remove_add_raindrops выполняет

raindrop.rect.y -= self.settings.screen_height

, чтобы изменить y атрибут Raindrop's rect.

Но это изменение немедленно перезаписывается, когда Raindrop 'update вызывается функция:

def update(self):
    """Shift raindrops to one level below"""
    self.y += self.settings.rain_speed
    self.rect.y = self.y

, потому что она снова изменяется на self.y.

Просто удалите _remove_add_raindrops (удаление и немедленное чтение Raindrop в self.raindrops не дает в любом случае) и измените ваш Raindrop класс примерно так:

class Raindrop(Sprite):
    """A class to represent a single raindrop."""

    def __init__(self, rain_game):
        """Initialize the raindrop and set its starting position."""
        super().__init__()
        self.screen = rain_game.screen
        self.settings = rain_game.settings

        # Load the raindrop image and set its rect attribute
        self.image = pygame.image.load('raindrop.bmp')
        self.rect = self.image.get_rect()

        #Start each raindrop near the top left of the screen.
        self.rect.x = self.rect.width
        self.rect.y = self.rect.height

    def update(self):
        """Shift raindrops to one level below"""
        self.rect.move_ip(0, self.settings.rain_speed)
        if not pygame.display.get_surface().get_rect().bottom > self.rect.top:
            self.rect.bottom = 0

, чтобы класс Raindrop мог сам проверить, выходит ли он из экрана. Если вы хотите сохранить положение Sprite с использованием чисел с плавающей точкой вместо целых чисел, я рекомендую использовать для этого класс Vector2 вместо отдельных атрибутов, поскольку это облегчает перемещение спрайтов по диагонали в долгосрочной перспективе.

0 голосов
/ 06 апреля 2020

Проблема заключалась в том, что self.y увеличивался бесконечно, поскольку я не переустанавливал его нигде в исходном коде или в обновленном коде.

def update(self):
    """Shift raindrops to one level below"""
    self.y += self.settings.rain_speed
    self.rect.y = self.y
    if self.rect.top >= self.screen.get_rect().bottom:
        self.rect.y -= self.settings.screen_height

Изменение последней строки выше от "self". rect.y "to" self.y "решил проблему.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...