I’m making a simple python game and there are 3 py files: alien_invasion, settings, ship. I would like the image position to be at the middle bottom of screen every time the game starts. It works when code are like the following:
JavaScriptx21alien_invasion.py:
2
JavaScript
1
56
56
1
import sys
2
import pygame
3
4
from settings import Settings
5
from ship import Ship
6
7
class AlienInvasion:
8
"""Overall class to manage game assets and behaviour."""
9
10
def __init__(self):
11
"""Initialize the game, and create game resources."""
12
pygame.init()
13
self.settings = Settings()
14
15
self.screen = pygame.display.set_mode(
16
(self.settings.screen_width, self.settings.screen_height), pygame.RESIZABLE)
17
pygame.display.set_caption("Alien Invasion")
18
19
self.ship = Ship(self)
20
21
def run_game(self):
22
"""Start the main loop for the game."""
23
while True:
24
self._check_events()
25
self.ship.update()
26
self._update_screen()
27
28
def _check_events(self):
29
"""Respond to keypresses and mouse events."""
30
for event in pygame.event.get():
31
if event.type == pygame.QUIT:
32
sys.exit()
33
elif event.type == pygame.KEYDOWN:
34
if event.key == pygame.K_RIGHT:
35
self.ship.moving_right = True
36
elif event.key == pygame.K_LEFT:
37
self.ship.moving_left = True
38
39
elif event.type == pygame.KEYUP:
40
if event.key == pygame.K_RIGHT:
41
self.ship.moving_right = False
42
elif event.key == pygame.K_LEFT:
43
self.ship.moving_left = False
44
45
def _update_screen(self):
46
""""Update images on the screen, and flip to the new screen."""
47
self.screen.fill(self.settings.bg_color)
48
self.ship.blitme()
49
50
pygame.display.flip()
51
52
if __name__ == '__main__':
53
# Make a game instance, and run the game.
54
ai = AlienInvasion()
55
ai.run_game()
56
ship.py:
JavaScript
1
33
33
1
import pygame
2
3
class Ship:
4
"""A class to manage the ship."""
5
6
def __init__(self, ai_game):
7
"""Initialize the ship and set its starting position."""
8
self.screen = ai_game.screen
9
self.screen_rect = ai_game.screen.get_rect()
10
11
# Load ship image and get its rect.
12
self.image = pygame.image.load('images/ship.bmp')
13
self.rect = self.image.get_rect()
14
15
# Start each new ship at the bottom center of the screen.
16
self.rect.midbottom = self.screen_rect.midbottom
17
18
# Movement flags
19
self.moving_right = False
20
self.moving_left = False
21
22
def update(self):
23
""""Update the ship's position based on the movement flags."""
24
if self.moving_right:
25
self.rect.x += 1
26
if self.moving_left:
27
self.rect.x -= 1
28
29
def blitme(self):
30
"""Draw the ship at its current location."""
31
self.screen.blit(self.image, self.rect)
32
33
settings.py:
JavaScript
1
10
10
1
class Settings:
2
"""A class to store all settings for Alien Invasion."""
3
4
def __init__(self):
5
"""Initialize the game's settings."""
6
# Screen settings
7
self.screen_width = 1200
8
self.screen_height = 800
9
self.bg_color = (230, 230, 230)
10
Then I made changes to settings.py and ship.py to adjust the moving speed:
ship.py:
JavaScript
1
40
40
1
import pygame
2
3
class Ship:
4
"""A class to manage the ship."""
5
6
def __init__(self, ai_game):
7
"""Initialize the ship and set its starting position."""
8
self.screen = ai_game.screen
9
self.settings = ai_game.settings
10
self.screen_rect = ai_game.screen.get_rect()
11
12
# Load ship image and get its rect.
13
self.image = pygame.image.load('images/ship.bmp')
14
self.rect = self.image.get_rect()
15
16
# Store a decimal value for the ship's horizontal position
17
self.x = float(self.rect.x)
18
19
# Start each new ship at the bottom center of the screen.
20
self.rect.midbottom = self.screen_rect.midbottom
21
22
# Movement flags
23
self.moving_right = False
24
self.moving_left = False
25
26
def update(self):
27
""""Update the ship's position based on the movement flags."""
28
## Update the ship's x value, not the rect.
29
if self.moving_right:
30
self.x += self.settings.ship_speed
31
if self.moving_left:
32
self.x -= self.settings.ship_speed
33
34
# Update rect object from self.x.
35
self.rect.x = self.x
36
37
def blitme(self):
38
"""Draw the ship at its current location."""
39
self.screen.blit(self.image, self.rect)
40
settings.py:
JavaScript
1
12
12
1
class Settings:
2
"""A class to store all settings for Alien Invasion."""
3
4
def __init__(self):
5
"""Initialize the game's settings."""
6
# Screen settings
7
self.screen_width = 1200
8
self.screen_height = 800
9
self.bg_color = (230, 230, 230)
10
# Ship settings
11
self.ship_speed = 1.5
12
Then when I run the alien_invasion code, the image starts showing up at the left bottom of screen when game starts, which is not what I want.
Advertisement
Answer
You need to set self.x
after setting the location of self.rect
:
JavaScript
1
18
18
1
class Ship:
2
"""A class to manage the ship."""
3
4
def __init__(self, ai_game):
5
# [...]
6
7
self.rect = self.image.get_rect()
8
9
# Store a decimal value for the ship's horizontal position
10
# self.x = float(self.rect.x) <--- delete
11
12
# Start each new ship at the bottom center of the screen.
13
self.rect.midbottom = self.screen_rect.midbottom
14
15
self.x = float(self.rect.x) # <--- insert
16
17
# [...]
18