#! /usr/bin/env python2 # The above line makes this executible in unix systems # Asteroids Infinity # Inspired by the classic Atari arcade game "Asteroids" # Copyright (C) 2009 Ben Whittaker # # This program is free software: you can redistribute it and/or modify # it under the terms of the GNU General Public License as published by # the Free Software Foundation, either version 3 of the License, or # (at your option) any later version. # # This program is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU General Public License for more details. # # You should have received a copy of the GNU General Public License # along with this program. If not, see . import pygame import random import math import os import sys import string from pygame.locals import * pygame.init() #joystick or gamepad init pygame.joystick.init() print ("Joystics: ", pygame.joystick.get_count()) js = pygame.joystick.Joystick(0) js.init() try: os.chdir(sys.argv[0][:sys.argv[0].rindex("AsteroidsInfinity.py")]) except: pass screensize = (640, 480) playarea = (700, 540) viewpoint = [350, 270] forecolor = (255, 255, 255) backcolor = (0, 0, 0) Objects = pygame.sprite.Group() ProtoObjs = pygame.sprite.Group() Saucers = pygame.sprite.Group() Asteroids = pygame.sprite.Group() Particles = pygame.sprite.Group() Collidable = pygame.sprite.Group() Bullets = pygame.sprite.Group() ShipGroup = pygame.sprite.GroupSingle() TextGroup = pygame.sprite.RenderUpdates() def loadsounds(): global explosion_sound global shot_sound global thrust_sound global beat1_sound global beat2_sound global life_sound global saucer_sound explosion_sound = Sound("sounds/explosion.wav") shot_sound = Sound("sounds/shot.wav") thrust_sound = Sound("sounds/thrust.wav") beat1_sound = Sound("sounds/beat1.wav") beat2_sound = Sound("sounds/beat2.wav") life_sound = Sound("sounds/life.wav") saucer_sound = Sound("sounds/saucer.wav") if pygame.mixer.get_init() <> None: pygame.mixer.set_reserved(2) # reserves channels for the thrust sound and the saucer sound pygame.mixer.Channel(1).set_volume(0.5) def set_volume(volume): explosion_sound.set_volume(volume) shot_sound.set_volume(volume) thrust_sound.set_volume(volume) beat1_sound.set_volume(volume) beat2_sound.set_volume(volume) life_sound.set_volume(volume) saucer_sound.set_volume(volume) def get_highscores(): if os.access("highscores.txt", os.F_OK) == True: try: scores_file = open("highscores.txt","r") highscores = [] lines = scores_file.readlines() scores_file.close() for line in lines: line = line.strip() line = line.split(":") if len(line) < 2: break highscores += [(int(line[0]), line[1])] while len(highscores) < 10: highscores += [(0,"")] return highscores except: print "Could not open highscores and/or could not parse highscores." print "Using default highscores." highscores = [ (8128, "PERFECT"), (6173, "PRIME"), (4540, "11BC"), (3798, "RAND"), (3141, "PI"), (2718, "E"), (1764, "LIFE^2"), (1024, "10^1010"), (525, "BAR"), (325, "FOO") ] return highscores def save_highscores(highscores): scores_file = open("highscores.txt","w") for score in highscores: scores_file.write(str(score[0]) + ":" + score[1] + "\n") scores_file.close() def get_controls(): controls = {"left":pygame.K_LEFT, "right":pygame.K_RIGHT, "up":pygame.K_UP, "down":pygame.K_DOWN, "shoot":pygame.K_SPACE, "shield":pygame.K_LCTRL} if os.access("controls.txt", os.F_OK) == True: controls_file = open("controls.txt", "r") for a in ("up", "down", "left", "right", "shoot", "shield"): line = controls_file.readline()[:-1] if line.isdigit() == True: controls[a] = int(line) controls_file.close() return controls def save_controls(controls): controls_file = open("controls.txt","w") for a in ("up", "down", "left", "right", "shoot", "shield"): controls_file.write(str(controls[a]) + "\n") controls_file.close() class DummySound(): def __init__(self, filename): pass def play(self, loops=0, maxtime=0, fade_ms=0): pass def stop(self): pass def fadeout(self, time): pass def set_volume(self, value): pass def get_volume(self): return 1 def get_num_channels(self): return 0 def get_length(self): return 0 def Sound(filename): if pygame.mixer.get_init() == None: return DummySound(filename) else: return pygame.mixer.Sound(filename) class DummyChannel(): def __init__(self, ID): pass def play(self, Sound, loops=0, maxtime=0, fade_ms=0): pass def stop(self): pass def set_volume(self, value): pass def get_volume(self): return 1 def get_busy(self): return False def queue(self, Sound): pass def Channel(ID): if pygame.mixer.get_init() == None: return DummyChannel(ID) else: return pygame.mixer.Channel(ID) class Text(pygame.sprite.Sprite): # A class of objects for displaying text def __init__(self, text, font, pos, align = "left"): pygame.sprite.Sprite.__init__(self, TextGroup) self.align = align # short for alignment, and can be "left", "right", or "center" self.pos = pos self.font = font self.change(text) def change(self, text): self.image = self.font.render(text, 1, (forecolor)) # make the text graphic for a in range(2): self.image.blit(self.image, (0, 0)) # copy the text graphic onto itself to make it brighter self.text = text width = self.image.get_width() height = self.image.get_height() if self.align == "left": self.rect = pygame.Rect(self.pos[0], self.pos[1], width, height) elif self.align == "center": self.rect = pygame.Rect(self.pos[0] - width / 2, self.pos[1], width, height) else: self.rect = pygame.Rect(self.pos[0] - width, self.pos[1], width, height) def wrap(num, start, end): while num < start: num += end - start while num > end: num -= end - start return num def explosion(pos, rel_speed, particles, added_speed = 200): for a in range(0, particles): angle = random.uniform(0, math.pi * 2) speed = random.uniform(0, added_speed) Particle(pos, (rel_speed[0] + math.sin(angle) * speed, rel_speed[1] + math.cos(angle) * speed)) explosion_sound.play() class Obj(pygame.sprite.Sprite): angle = 0 spin = 0 def __init__(self, pos, speed, groups = []): pygame.sprite.Sprite.__init__(self, groups + [Objects]) self.pos = list(pos) self.speed = list(speed) self.set_pos_screen() #self.image = image #self.image.set_colorkey((66, 66, 66)) #self.rect = pygame.Rect((self.pos[0] - self.image.get_width() / 2, self.pos[1] - self.image.get_height() / 2), image.get_size()) def update(self): self.angle += self.spin / fps self.pos[0] += self.speed[0] / fps self.pos[1] += self.speed[1] / fps if self.pos[0] > playarea[0]: self.pos[0] -= playarea[0] elif self.pos[0] < 0: self.pos[0] += playarea[0] if self.pos[1] > playarea[1]: self.pos[1] -= playarea[1] elif self.pos[1] < 0: self.pos[1] += playarea[1] self.set_pos_screen() #self.pos_screen = (self.rect.x + self.rect.w / 2, self.rect.y + self.rect.h / 2) ## def render(self): ## render_pos = (self.image.get_height() / 2, self.image.get_width() / 2) ## temp_points = [] ## for point in self.image_points: ## temp_points.append((math.sin(point[0] + self.angle) * point[1] + render_pos[0], math.cos(point[0] + self.angle) * point[1] + render_pos[1])) ## self.image.fill((66, 66, 66)) ## pygame.draw.polygon(self.image, (0, 0, 0), temp_points) ## pygame.draw.aalines(self.image, (255, 255, 255), 1, temp_points) ## ## #pygame.draw.circle(self.image, (100, 100, 100), render_pos, self.radius, 1) def set_pos_screen(self): self.pos_screen = [self.pos[0] - viewpoint[0], self.pos[1] - viewpoint[1]] max_radius_x = (playarea[0] - screensize[0]) / 2 max_radius_y = (playarea[1] - screensize[1]) / 2 #if self.pos_screen[0] - max_radius_x > screensize[0]: # self.pos_screen[0] -= playarea[0] #elif self.pos_screen[0] + max_radius_x < 0: # self.pos_screen[0] += playarea[0] #if self.pos_screen[1] - max_radius_y > screensize[1]: # self.pos_screen[1] -= playarea[1] #elif self.pos_screen[1] + max_radius_y < 0: # self.pos_screen[1] += playarea[1] self.pos_screen[0] = wrap(self.pos_screen[0], -max_radius_x, screensize[0] + max_radius_x) self.pos_screen[1] = wrap(self.pos_screen[1], -max_radius_y, screensize[1] + max_radius_y) def draw(self, surface): render_pos = (self.pos_screen[0], self.pos_screen[1]) dirtyrects = [] for points_list in self.image_points: color = forecolor if len(points_list[0]) > 2: color = points_list[0] points_list = points_list[1:] temp_points = [] for point in points_list: temp_points.append((math.sin(point[0] + self.angle) * point[1] + render_pos[0], math.cos(point[0] + self.angle) * point[1] + render_pos[1])) pygame.draw.polygon(surface, backcolor, temp_points) dirtyrects.append(pygame.draw.aalines(surface, color, 1, temp_points).inflate(2, 2)) #pygame.draw.circle(surface, (100, 100, 100), render_pos, self.radius, 1) return dirtyrects class Asteroid(Obj): def __init__(self, pos, speed, size, groups = []): self.size = size self.radius = 3 * pow(2, size) self.spin = random.uniform(math.pi * -2, math.pi * -2) #image = pygame.Surface((self.radius * 4, self.radius * 4)) self.image_points = [[]] for a in range(0, 10): angle = a * math.pi / 5 + random.uniform(0, math.pi / 5) dist = random.uniform(self.radius * 0.7, self.radius * 1.4) self.image_points[0].append((angle, dist)) Obj.__init__(self, pos, speed, groups + [Asteroids, Collidable]) self.collision_type = "hard" #self.render() def collide(self, victim): global score if victim.collision_type == "Explosive": #for a in range(50): # angle = random.uniform(0, math.pi * 2) # speed = random.uniform(0, 10) # Particle(self.pos, (self.speed[0] + math.sin(angle) * speed, self.speed[1] + math.cos(angle) * speed)) explosion(self.pos, self.speed, 50) for a in range(2): if self.size > 1: angle = random.uniform(0, math.pi * 2) speed = random.uniform(0, 100) Asteroid(self.pos, (self.speed[0] + math.sin(angle) * speed, self.speed[1] + math.cos(angle) * speed), self.size - 1) raise_score(25 * pow(2, 3 - self.size), victim) if random.randint(0, 50) == 0: if random.randint(1, 3) == 1: SmallSaucer((random.randint(0, screensize[0]), random.randint(0, screensize[1]))) else: BigSaucer((random.randint(0, screensize[0]), random.randint(0, screensize[1]))) self.kill() class Ship(Obj): def __init__(self, pos, speed, radius, movement_keys, groups = []): self.radius = radius self.movement_keys = movement_keys #image = pygame.Surface((self.radius * 4, self.radius * 4)) ## self.image_points = [[ ## (0, radius * 1.3), ## (math.pi / 3, radius), ## (math.pi * 5 / 6, radius * 1.2), ## (math.pi, radius / 2), ## (math.pi * 7 / 6, radius * 1.2), ## (math.pi * 5 / 3, radius) ## ]] self.normal_points = [[ (0, radius * 1.3), (math.pi / 2, radius / 2), (math.pi / 2, radius * 1.2), (math.pi * 5 / 6, radius), (math.pi * 7 / 6, radius), (math.pi * 3 / 2, radius * 1.2), (math.pi * 3 / 2, radius / 2) ]] self.shield_points = [(math.pi / 10 * a, self.radius * 1.5) for a in range(20)] self.image_points = self.normal_points Obj.__init__(self, pos, speed, groups + [Collidable]) self.collision_type = "Explosive" #self.render() self.gunheat = 0 self.reloading = 0 self.bullets = pygame.sprite.Group() self.activate_shield(2) self.shield = 5 def control(self, keys): self.thrustvolume = 0 self.spin = 0 factor = 1 if keys[self.movement_keys["left"]] == 1: self.spin += factor*math.pi if random.random() < 20 / fps: self.exaust(math.pi, (math.pi * 1 / 2, self.radius * 1.2), 1) self.thrustvolume = 0.25 if keys[self.movement_keys["right"]] == 1: self.spin -= factor*math.pi if random.random() < 20 / fps: self.exaust(math.pi, (math.pi * 3 / 2, self.radius * 1.2), 1) self.thrustvolume = 0.25 if keys[self.movement_keys["down"]] == 1: self.speed[0] -= math.sin(self.angle) * 400 / 3 / fps self.speed[1] -= math.cos(self.angle) * 400 / 3 / fps if random.random() < 20 / fps: self.exaust(math.pi, (math.pi * 1 / 2, self.radius * 1.2), 1) self.exaust(math.pi, (math.pi * 3 / 2, self.radius * 1.2), 1) self.thrustvolume = 0.75 if keys[self.movement_keys["up"]] == 1: self.speed[0] += factor*math.sin(self.angle) * 800 / 3 / fps self.speed[1] += factor*math.cos(self.angle) * 800 / 3 / fps if random.random() < 60 / fps: self.exaust(0, (0, -self.radius / 2), 1) self.thrustvolume = 1 Channel(0).set_volume(self.thrustvolume) Channel(0).queue(thrust_sound) if keys[self.movement_keys["shoot"]] == 1 and self.gunheat < 40 and self.reloading <= 0: Bullet(self.pos, (self.speed[0] + math.sin(self.angle) * 400, self.speed[1] + math.cos(self.angle) * 400), self.angle, 4, creator = self, groups = [self.bullets]) self.gunheat += 10 self.reloading += 1 if self.reloading > 0: self.reloading -= 20.0 / fps if self.gunheat > 0: self.gunheat -= 20.0 / fps if self.start_shield > 0: self.start_shield -= 1.0 / fps elif self.collision_type == "Hard": self.collision_type = "Explosive" self.image_points = self.normal_points if keys[self.movement_keys["shield"]] == 1 and self.shield > 0: if self.shield > 5: self.shield = 5.0 self.collision_type = "Hard" s = self.shield / 5 self.image_points = [[[int(s * forecolor[n] + (1 - s) * backcolor[n]) for n in (0,1,2)]] + self.shield_points] + self.normal_points self.shield -= 2.5 / fps elif self.shield < 5: self.shield += 0.5 / fps self.collision_type = "Explosive" self.image_points = self.normal_points def exaust(self, rel_angle, rel_pos, amount): for a in range(amount): angle = self.angle + rel_angle + random.uniform(math.pi / -12, math.pi / 12) speed = random.uniform(100, 200) pos = [self.pos[0] + math.sin(rel_pos[0] + self.angle) * rel_pos[1], self.pos[1] + math.cos(rel_pos[0] + self.angle) * rel_pos[1]] Particle(pos, (self.speed[0] + math.sin(angle) * -speed, self.speed[1] + math.cos(angle) * -speed)) def activate_shield(self, timer): self.start_shield = timer self.collision_type = "Hard" self.image_points = [self.shield_points] + self.normal_points def collide(self, victim): if victim not in self.bullets.sprites() and self.collision_type == "Explosive": for a in range(0, 7): angle = random.uniform(0, math.pi * 2) speed = random.uniform(0, 200) Stick(self.pos, (self.speed[0] + math.sin(angle) * speed, self.speed[1] + math.cos(angle) * speed), 7) explosion(self.pos, self.speed, 25) self.kill() class Saucer(Obj): def __init__(self, pos, radius, groups = []): Obj.__init__(self, pos, [0, 0], groups + [Particles, ProtoObjs, Saucers]) self.remove(Objects) self.real_radius = radius self.radius = 0.5 self.make_image_points(self.radius) self.collision_type = "Explosive" angle = random.uniform(0, math.pi * 2) speed = random.randint(40, 160) self.speed[0] += math.sin(angle) * speed self.speed[1] += math.cos(angle) * speed self.bullets = pygame.sprite.Group() if Saucers.sprites() == [self]: #print 1 Channel(1).play(saucer_sound, -1) #else: #print 0 def update(self): global viewpoint if self.radius < self.real_radius: self.radius *= 7**(1 / fps) self.make_image_points(self.radius) else: self.radius = self.real_radius self.remove(ProtoObjs) self.add(Objects) self.add(Collidable) if random.random() > 0.35**(1/fps): self.speed = list(viewpointspeed) angle = random.uniform(0, math.pi * 2) speed = random.randint(40, 160) self.speed[0] += math.sin(angle) * speed self.speed[1] += math.cos(angle) * speed # The following commented out chunk off code was an attempt to make saucers avoid asteroids. It didn't work. ## for a in range(10): ## self.speed = list(viewpointspeed) ## angle = random.uniform(0, math.pi * 2) ## speed = random.randint(40, 160) ## self.speed[0] += math.sin(angle) * speed ## self.speed[1] += math.cos(angle) * speed ## danger = 0 ## for asteroid in Asteroids.sprites(): ## rel_speed = (asteroid.speed[0] - self.speed[0], asteroid.speed[1] - self.speed[1]) ## rel_pos = (wrap(asteroid.pos[0] - self.pos[0], -playarea[0]/2, playarea[0]/2), wrap(asteroid.pos[1] - self.pos[1], -playarea[1]/2, playarea[1]/2)) ## rel_angle = math.atan2(rel_pos[0], rel_pos[1]) - 180 ## clearance = math.atan2(asteroid.radius + self.radius, math.hypot(rel_pos[0], rel_pos[1])) ## #print clearance ## ras = math.atan2(rel_speed[0], rel_speed[1]) ## #print ras ## #angle = rel_angle ## #speed = 200 ## #Stick(self.pos, (self.speed[0] + math.sin(angle) * speed, self.speed[1] + math.cos(angle) * speed), 5) ## if wrap(rel_angle - clearance, ras - math.pi, ras + math.pi) < ras < wrap(rel_angle + clearance, ras - math.pi, ras + math.pi): ## danger = 1 ## #print "not there" ## if a == 9: ## print "Aaaaaaaa!!!!!" ## break ## if danger == 0: ## break Obj.update(self) def collide(self, victim): if victim not in self.bullets.sprites(): explosion(self.pos, self.speed, 25) for a in range(0, 10): angle = random.uniform(0, math.pi * 2) speed = random.uniform(0, 200) Stick(self.pos, (self.speed[0] + math.sin(angle) * speed, self.speed[1] + math.cos(angle) * speed), self.radius / 2) self.kill() def make_image_points(self, r): self.image_points = [[ (math.pi / 2, r * 4 / 3), (math.pi / 4, r * 3 / 4), (-math.pi / 4, r * 3 / 4), (-math.pi / 2, r * 4 / 3) ], [ (math.pi * 3 / 4, r * 3 / 4), (math.pi / 2, r * 4 / 3), (-math.pi / 2, r * 4 / 3), (-math.pi * 3 / 4, r * 3 / 4) ], [ (math.pi * 11 / 12, r), (math.pi * 3 / 4, r * 3 / 4), (-math.pi * 3 / 4, r * 3 / 4), (-math.pi * 11 / 12, r) ]] def kill(self): pygame.sprite.Sprite.kill(self) if Saucers.sprites() == []: Channel(1).stop() class BigSaucer(Saucer): def __init__(self, pos): Saucer.__init__(self, pos, 15) def update(self): Saucer.update(self) if random.random() > 0.6**(1/fps) and Objects in self.groups(): angle = random.uniform(0, math.pi * 2) speed = (self.speed[0] + math.sin(angle) * 400, self.speed[1] + math.cos(angle) * 400) Bullet([self.pos[0], self.pos[1]], speed, angle, 4, creator = self, groups = [self.bullets]) def collide(self, victim): global score Saucer.collide(self, victim) raise_score(250, victim) class SmallSaucer(Saucer): def __init__(self, pos): Saucer.__init__(self, pos, 10) def update(self): Saucer.update(self) if ShipGroup.sprite <> None: target = ShipGroup.sprite elif Asteroids.sprites() <> []: target = Asteroids.sprites()[0] if random.random() > 0.6**(1/fps) and Objects in self.groups(): angle = math.atan2(target.pos_screen[0] - self.pos_screen[0], target.pos_screen[1] - self.pos_screen[1]) + random.uniform(math.pi / -4, math.pi / 4) speed = (self.speed[0] + math.sin(angle) * 400, self.speed[1] + math.cos(angle) * 400) Bullet([self.pos[0], self.pos[1]], speed, angle, 4, creator = self, groups = [self.bullets]) def collide(self, victim): global score Saucer.collide(self, victim) raise_score(1000, victim) class Particle(Obj): def __init__(self, pos, speed, groups = []): #image = pygame.Surface((1, 1)) #image.fill((200, 200, 200)) Obj.__init__(self, pos, speed, groups + [Particles]) def update(self): if random.random() > 0.35**(1/fps): self.kill() Obj.update(self) def draw(self, surface): pos_screen_int = [int(self.pos_screen[0]), int(self.pos_screen[1])] surface.set_at(pos_screen_int, forecolor) return [pygame.Rect(pos_screen_int, (1, 1))] class Stick(Obj): def __init__(self, pos, speed, radius, groups = []): self.radius = radius #image = pygame.Surface((radius * 2, radius * 2)) self.spin = random.uniform(math.pi * -2, math.pi * 2) self.image_points = [[(0, 0), (0, radius), (math.pi, radius)]] Obj.__init__(self, pos, speed, groups + [Particles]) def update(self): if random.random() > 0.50**(1/fps): self.kill () Obj.update(self) class Bullet(Obj): def __init__(self, pos, speed, angle, radius, creator = None, groups = []): self.creator = creator self.radius = radius self.angle = angle #image = pygame.Surface((radius * 2, radius * 2)) self.image_points = [[(0, radius), (math.pi * 8 / 9, radius), (math.pi * 10 / 9, radius)]] self.collision_type = "Explosive" Obj.__init__(self, pos, speed, groups + [Collidable, Bullets]) #self.render() self.lifetime = 1 shot_sound.play() def update(self): self.lifetime -= 1.0 / fps if self.lifetime <= 0: self.kill() Obj.update(self) def collide(self, victim): if victim <> self.creator: if not (Bullets in victim.groups() and victim.creator == self.creator): explosion(self.pos, [(self.speed[a] + victim.speed[a]) / 2 for a in range(2)], 25, 100) self.kill() def raise_score(amount, attacker = None): global score if attacker == None: score += amount elif attacker == ShipGroup.sprite: score += amount elif attacker.__class__ == Bullet and attacker.creator == ShipGroup.sprite: score += amount def main(): global viewpoint global viewpointspeed global score global fps font = pygame.font.Font("font/Vectorb.ttf", 20) small_font = pygame.font.Font("font/Vectorb.ttf", 10) large_font = pygame.font.Font("font/Vectorb.ttf", 40) loadsounds() screen = pygame.display.set_mode(screensize,FULLSCREEN|DOUBLEBUF|HWSURFACE) screen.fill(backcolor) pygame.display.flip() pygame.mouse.set_visible(False) fullscreen = 0 pygame.display.set_caption("Asteroids Infinity") volume = 100 clock = pygame.time.Clock() highscores = get_highscores() controls = get_controls() # Asteroid([random.randint(0, 1000), random.randint(0, 1000)], (random.randint(-5, 5), random.randint(-5, 5)), 3) #Ship([playarea[0] / 2, playarea[1] / 2], [0, 0], 10, {"left":pygame.K_LEFT, "right":pygame.K_RIGHT, "up":pygame.K_UP, "down":pygame.K_DOWN, "shoot":pygame.K_SPACE, "shield":pygame.K_LCTRL}, groups = [ShipGroup]) viewpointspeed = [0, 0] score = 0 lives = 0 #lives_text = font.render("Lives : " + str(lives), 1, (255, 255, 255)) lives_text = Text("Lives : " + str(lives), font, (20, 20)) level_text = Text("Level N/A", font, (screensize[0] / 2, 20), align = "center") score_text = Text("", font, (screensize[0] - 20, 20), align = "right") fps_counter = 0 respawn_time = 2 running = 1 for a in range(random.randrange(1, 10)): Asteroid((random.randint(0, playarea[0]), random.randint(0, playarea[1])), (random.randint(-100, 100), random.randint(-100, 100)), random.randint(1, 3)) lastbeat = pygame.time.get_ticks() beattype = 1 mode = "menustart" olddirtyrects = [] #for joystick moves k_left = 0 k_right = 0 k_up = 0 k_down = 0 k_shoot = 0 k_shield = 0 while running == 1: fps = 1000.0 / clock.tick(100) if fps_counter == 1: fps_text.change("FPS : " + str(clock.get_fps())) keys = pygame.key.get_pressed() #if keys[pygame.K_ESCAPE] == True: # break events = pygame.event.get() lst = list(keys) #A if (js.get_button(1) == 1): lst[controls["shoot"]]= 1 k_shoot = 1 else: k_shoot = 0 #B if (js.get_button(0) == 1): lst[controls["shield"]] = 1 k_shield = 1 #left if (js.get_axis(0) < 0 ): lst[controls["left"]] = 1 k_left = 1 #right if (js.get_axis(0) > 0 ): lst[controls["right"]] = 1 k_right = 1 #up if (js.get_axis(1) < 0 ): lst[controls["up"]] = 1 #down if (js.get_axis(1) > 0 ): lst[controls["down"]] = 1 keys = tuple(lst) for event in events: if event.type == pygame.QUIT: running = 0 elif event.type == JOYBUTTONDOWN: # Button A if (js.get_button(1) == 1): if mode == "menu": if selected == 0: for item in menu_items: item.kill() mode = "playstart" elif event.type == pygame.KEYDOWN: #print event.unicode if event.key == pygame.K_ESCAPE: if mode == "play": mode = "gameoverstart" elif mode == "menu": running = 0 elif mode == "controls" or mode == "setcontrol": for text in controls_text + menu_items: text.kill() mode = "optionsstart" elif mode <> "gameover": for text in menu_items: text.kill() if mode == "sethighscore": name_text.kill() mode = "menustart" if event.key == pygame.K_p and mode == "play": # Pause pause_text1 = Text("PAUSED", large_font, (screensize[0] / 2, screensize[1] / 2 - 65), align = "center") pause_text2 = Text("Press any key to continue", font, (screensize[0] / 2, screensize[1] / 2 + 20), align = "center") TextGroup.draw(screen) pygame.display.flip() while 1: event = pygame.event.wait() if event.type == pygame.KEYDOWN: pause_text1.kill() pause_text2.kill() break if event.type == pygame.QUIT: running = 0 break clock.tick() if mode == "menu" or mode == "options" or mode == "controls": if event.key == pygame.K_UP or event.key == pygame.K_DOWN: menu_items[selected].change(menu_items[selected].text[1:-1]) if event.key == pygame.K_UP: if selected == 0: selected = len(menu_items) - 1 else: selected -= 1 else: if selected == len(menu_items) - 1: selected = 0 else: selected += 1 menu_items[selected].change("<" + menu_items[selected].text + ">") if mode == "menu": if event.key == pygame.K_RETURN: if selected == 0: for item in menu_items: item.kill() mode = "playstart" elif selected == 1: for item in menu_items: item.kill() mode = "highscoresstart" elif selected == 2: for item in menu_items: item.kill() mode = "optionsstart" elif selected == 3: running = 0 #title_text.kill() if mode == "options": if event.key == pygame.K_RETURN: if selected == 0: if fullscreen == 0: pygame.display.set_mode(screensize, pygame.FULLSCREEN) pygame.mouse.set_visible(False) screen.fill((backcolor)) pygame.display.flip() fullscreen = 1 menu_items[0].change("") else: pygame.display.set_mode(screensize) pygame.mouse.set_visible(True) screen.fill((backcolor)) pygame.display.flip() fullscreen = 0 menu_items[0].change("") elif selected == 2: if fps_counter == 0: fps_counter = 1 fps_text = Text("", small_font, (2, screensize[1] - 13)) menu_items[2].change("") else: fps_counter = 0 fps_text.kill() menu_items[2].change("") elif selected == 3: for item in menu_items: item.kill() mode = "controlsstart" elif selected == 4: for item in menu_items: item.kill() mode = "menustart" if selected == 1: if event.key == pygame.K_LEFT and volume > 0: volume -= 10 set_volume(volume / 100.0) menu_items[1].change("") elif event.key == pygame.K_RIGHT and volume < 100: volume += 10 set_volume(volume / 100.0) menu_items[1].change("") if mode == "controls": if event.key == pygame.K_RETURN: if selected < 6: menu_items[selected].change("< >") mode = "setcontrol" if selected == 6: for text in controls_text + menu_items: text.kill() mode = "optionsstart" elif mode == "setcontrol": menu_items[selected].change("<" + pygame.key.name(event.key) + ">") if selected == 0: controls["up"] = event.key elif selected == 1: controls["down"] = event.key elif selected == 2: controls["left"] = event.key elif selected == 3: controls["right"] = event.key elif selected == 4: controls["shoot"] = event.key elif selected == 5: controls["shield"] = event.key save_controls(controls) mode = "controls" if mode == "sethighscore": if event.unicode in string.printable[:95] and len(name) < 10: name += event.unicode name_text.change(name + "_") elif event.key == pygame.K_BACKSPACE: name = name[:-1] name_text.change(name + "_") elif event.key == pygame.K_RETURN: highscores += [(score, name.upper())] highscores.sort(reverse = True) highscores = highscores[:10] save_highscores(highscores) for text in menu_items: text.kill() name_text.kill() mode = "highscoresstart" if mode == "gameover": for text in gameover_text: text.kill() mode = "highscorecheck" if mode == "highscores": for text in menu_items: text.kill() mode = "menustart" if event.type == pygame.MOUSEBUTTONDOWN and event.pos == (0, 0): pygame.display.set_mode(screensize) pygame.mouse.set_visible(True) fullscreen = 0 if mode == "options": menu_items[0].change("") #pygame.event.pump() a = 0 if Asteroids.sprites() == []: if mode == "play": level += 1 for obj in Objects.sprites(): obj.speed[0] -= viewpointspeed[0] obj.speed[1] -= viewpointspeed[1] for a in range(level): Asteroid([random.randint(0, playarea[0]), random.randint(0, playarea[1])], (random.randint(-100, 100), random.randint(-100, 100)), 3) #level_text = font.render("Level " + str(level), 1, (255, 255, 255)) level_text.change("Level " + str(level)) if ShipGroup.sprite <> None: ShipGroup.sprite.activate_shield(2) viewpointspeed = [0, 0] else: for a in range(random.randrange(1, 10)): Asteroid((random.randint(0, playarea[0]), random.randint(0, playarea[1])), (random.randint(-100, 100), random.randint(-100, 100)), random.randint(1, 3)) viewpointspeed = [0, 0] while a < len(Collidable.sprites()): obj1 = Collidable.sprites()[a] pos1 = list(obj1.pos) for obj2 in Collidable.sprites()[a + 1:]: pos2 = list(obj2.pos) if pos1[0] - pos2[0] > playarea[0] / 2: pos1[0] -= playarea[0] if pos2[0] - pos1[0] > playarea[0] / 2: pos2[0] -= playarea[0] if pos1[1] - pos2[1] > playarea[1] / 2: pos1[1] -= playarea[1] if pos2[1] - pos1[1] > playarea[1] / 2: pos2[1] -= playarea[1] if math.hypot(pos1[0] - pos2[0], pos1[1] - pos2[1]) < obj1.radius + obj2.radius: obj1.collide(obj2) obj2.collide(obj1) if obj1.groups() == []: break a += 1 if mode == "menustart": #title_text = Text("ASTEROIDS INFINITY", large_font, (screensize[0] / 2, screensize[1] / 2 - 180), align = "center") menu_items = ( Text("", large_font, (screensize[0] / 2, screensize[1] / 2 - 80), align = "center"), Text("HIGHSCORES", large_font, (screensize[0] / 2, screensize[1] / 2), align = "center"), Text("OPTIONS", large_font, (screensize[0] / 2, screensize[1] / 2 + 80), align = "center"), Text("QUIT", large_font, (screensize[0] / 2, screensize[1] / 2 + 160), align = "center") ) selected = 0 mode = "menu" if mode == "optionsstart": if fullscreen == 0: item0 = "" else: item0 = "" item1 = "VOLUME: " + str(volume) + "%" if fps_counter == 0: item2 = "FPS COUNTER ON" else: item2 = "FPS COUNTER OFF" menu_items = ( Text(item0, large_font, (screensize[0] / 2, screensize[1] / 2 - 180), align = "center"), Text(item1, large_font, (screensize[0] / 2, screensize[1] / 2 - 100), align = "center"), Text(item2, large_font, (screensize[0] / 2, screensize[1] / 2 - 20), align = "center"), Text("CONTROLS", large_font, (screensize[0] / 2, screensize[1] / 2 + 60), align = "center"), Text("BACK", large_font, (screensize[0] / 2, screensize[1] / 2 + 140), align = "center") ) selected = 0 mode = "options" if mode == "controlsstart": controls_text = ( Text("UP: ", font, (screensize[0] / 2, screensize[1] / 2 - 140), align = "right"), Text("DOWN: ", font, (screensize[0] / 2, screensize[1] / 2 - 100), align = "right"), Text("LEFT: ", font, (screensize[0] / 2, screensize[1] / 2 - 60), align = "right"), Text("RIGHT: ", font, (screensize[0] / 2, screensize[1] / 2 - 20), align = "right"), Text("SHOOT: ", font, (screensize[0] / 2, screensize[1] / 2 + 20), align = "right"), Text("SHIELD: ", font, (screensize[0] / 2, screensize[1] / 2 + 60), align = "right") ) menu_items = ( Text("<" + pygame.key.name(controls["up"]) + ">", font, (screensize[0] / 2, screensize[1] / 2 - 140), align = "left"), Text(pygame.key.name(controls["down"]), font, (screensize[0] / 2, screensize[1] / 2 - 100), align = "left"), Text(pygame.key.name(controls["left"]), font, (screensize[0] / 2, screensize[1] / 2 - 60), align = "left"), Text(pygame.key.name(controls["right"]), font, (screensize[0] / 2, screensize[1] / 2 - 20), align = "left"), Text(pygame.key.name(controls["shoot"]), font, (screensize[0] / 2, screensize[1] / 2 + 20), align = "left"), Text(pygame.key.name(controls["shield"]), font, (screensize[0] / 2, screensize[1] / 2 + 60), align = "left"), Text("BACK", large_font, (screensize[0] / 2, screensize[1] / 2 + 100), align = "center") ) selected = 0 setting_key = 0 mode = "controls" if mode == "highscoresstart": menu_items = [Text("HIGHSCORES", large_font, (screensize[0] / 2, 40), align = "center")] a = -130 n = 1 for highscore in highscores: menu_items.append(Text(str(n) + ". ", font, (200, screensize[1] / 2 + a), align = "right")) menu_items.append(Text(highscore[1], font, (200, screensize[1] / 2 + a), align = "left")) menu_items.append(Text(str(highscore[0]), font, (screensize[0] - 150, screensize[1] / 2 + a), align = "right")) a += 30 n += 1 menu_items.append(Text("PRESS ANY KEY TO CONTINUE", font, (screensize[0] / 2, screensize[1] - 60), align = "center")) mode = "highscores" if mode == "highscorecheck": if score > highscores[-1][0]: menu_items = [] menu_items.append(Text("YOU SET A NEW HIGHSCORE WITH " + str(score) + " POINTS", font, (screensize[0] / 2, screensize[1] / 2 - 40), align = "center")) menu_items.append(Text("PLEASE ENTER YOUR NAME", font, (screensize[0] / 2, screensize[1] / 2), align = "center")) name_text = Text("_", font, (screensize[0] / 2, screensize[1] / 2 + 40), align = "center") name = "" mode = "sethighscore" else: mode = "highscoresstart" if mode == "playstart": for sprite in Collidable.sprites(): sprite.kill() Ship([random.randint(0, playarea[0]), random.randint(0, playarea[0])], [0, 0], 10, controls, groups = [ShipGroup]) viewpoint = [playarea[0] / 2 - screensize[0] /2, playarea[1] / 2 - screensize[1] /2] viewpointspeed = [0, 0] score = 0 lives = 4 lives_text.change("Lives : " + str(lives)) level = 0 next_life = 10000 mode = "play" if mode == "play": if score >= next_life: lives += 1 lives_text.change("Lives : " + str(lives)) life_sound.play() next_life += 10000 if ShipGroup.sprite <> None: ship = ShipGroup.sprite ship.control(keys) #viewpoint[0] = ship.pos[0] - screensize[0] / 2 + ship.speed[0] / fps #viewpoint[1] = ship.pos[1] - screensize[1] / 2 + ship.speed[1] / fps rel_x = wrap(viewpoint[0] - ship.pos[0], -playarea[0], 0) + screensize[0] / 2 rel_y = wrap(viewpoint[1] - ship.pos[1], -playarea[1], 0) + screensize[1] / 2 px = math.fabs(rel_x / (screensize[0] / 8)) if px > 1: px = 1.0 py = math.fabs(rel_y / (screensize[1] / 8)) if py > 1: py = 1.0 viewpointspeed[0] -= (viewpointspeed[0] - ship.speed[0]) * (1 - ((1-px)**(1 / fps))) viewpointspeed[1] -= (viewpointspeed[1] - ship.speed[1]) * (1 - ((1-py)**(1 / fps))) p = (1 - (0.1**(1 / fps))) viewpoint[0] += -rel_x * p viewpoint[1] += -rel_y * p viewpoint[0] = wrap(viewpoint[0], 0, playarea[0]) viewpoint[1] = wrap(viewpoint[1], 0, playarea[1]) #if not -screensize[0] / 4 < rel_x < screensize[0] / 4: # viewpointspeed[0] = ship.speed[0] #if not -screensize[1] / 4 < rel_y < screensize[1] / 4: # viewpointspeed[1] = ship.speed[1] elif lives == 0: mode = "gameoverstart" elif respawn_time <= 0: Ship([random.randint(0, playarea[0]), random.randint(0, playarea[1])], [0, 0], 10, controls, groups = [ShipGroup]) lives -= 1 #lives_text = font.render("Lives : " + str(lives), 1, (255, 255, 255)) lives_text.change("Lives : " + str(lives)) respawn_time = 2 else: respawn_time -= 1.0 / fps else: if random.random() > 0.98**(1/fps): if random.randint(1, 3) == 1: SmallSaucer((random.randint(0, screensize[0]), random.randint(0, screensize[1]))) else: BigSaucer((random.randint(0, screensize[0]), random.randint(0, screensize[1]))) if mode == "gameoverstart": if ShipGroup.sprite <> None: ShipGroup.sprite.kill() gameover_text = ( Text("GAME OVER", large_font, (screensize[0] / 2, screensize[1] / 2 - 40), align = "center"), Text("Press any key to continue", font, (screensize[0] / 2, screensize[1] / 2 + 10), align = "center") ) mode = "gameover" if pygame.time.get_ticks() >= lastbeat + 10000 / (len(Asteroids.sprites()) + 10): lastbeat = pygame.time.get_ticks() if beattype == 1: beat1_sound.play() beattype = 2 else: beat2_sound.play() beattype = 1 viewpoint[0] += viewpointspeed[0] / fps viewpoint[1] += viewpointspeed[1] / fps screen.fill((backcolor)) Objects.update() ProtoObjs.update() #Objects.draw(screen) dirtyrects = [] for sprite in ProtoObjs.sprites(): dirtyrects += sprite.draw(screen) for sprite in Objects.sprites(): dirtyrects += sprite.draw(screen) #screen.blit(level_text, (screensize[0] / 2 - level_text.get_width() / 2, 5)) #score_text = font.render(str(score), 1, (255, 255, 255)) score_text.change(str(score)) #screen.blit(score_text, (screensize[0] - score_text.get_width() - 5, 5)) #screen.blit(lives_text, (5, 5)) dirtyrects += TextGroup.draw(screen) pygame.display.update(dirtyrects + olddirtyrects) olddirtyrects = dirtyrects main() pygame.quit()