首页 > 代码库 > Rice Rock

Rice Rock

先翻译评分要点,然后一点点翻译程序实现过程

如何产生一堆岩石?
rock_group = set([])#空集合,全局变量
 
rock_group.add(a_rock)
要画出来draw handler?
用rock_group代替a_rock,因为要产生一组岩石,有很多a_rock,代替哪一个呢?
在draw handler里面我用:
for n in rock_group:
    n.draw(canvea)
可以产生一堆岩石了,但是岩石后来就不转了,而且岩石数目一直增加-------用timer.stop()解决了
 process_sprite_group
看论坛,找到了答案
 
现在做撞击,看论坛,找到了答案。
如何产生一组导弹,岩石碰撞后不能再生
已经可以产生一组导弹了,但是导弹寿命要调整
碰撞后岩石消失,但是分数没有增加,岩石消失不再出现
分数已经增加啦,这个我自己做出来的
 
https://class.coursera.org/interactivepython-005/forum/thread?thread_id=5185
如何生成的导弹不碰到飞船
完成啦啦啦

Mini-project description - RiceRocks (Asteroids)

 

开发过程

 要有多重火箭和多重导弹。飞船撞到岩石会丢一条命, 导弹打到岩石得一分, 你要更新得分和生命情况,在合适的时候结束游戏。 .当发生碰撞的时候你可以选择添加爆炸动画

Tip #1 - Spawning rocks多重岩石

为了控制多重岩石,用rock_group代替 a_rock.在rock_spawner里面,你应该产生一个本地的a_rock的副本并添加到全局集合rock_group里面。 为了避免超过12个岩石,你可以使用一个基于rock_group长度的测试。 为了测试你的代码,你可以添加一个for循环到draw handler里面以这种形式: 对每个在rock_group里面的岩石, 画出那个岩石。这个draw以后将被放到 process_sprite_group里面在第一部分的最后一步

Phase one -  多重岩石

 

  1. 用rock_group代替a_rock.复位岩石组成空的集合set. 你的rock spawner产生一个新岩石(Sprite的一个对象实例) 并添加到rock_group里面
  2.  修改你的rock spawner使得屏幕里面的岩石数量有上限。我们建议最多12个太多岩石的话游戏会不那么好玩而且动画会明显变慢。
  3. 建立一个process_sprite_group函数. 这个函数应该有一个集合(set)和画布, 并调用update和draw函数对每个群组里面的sprite.
  4. 在draw handler里面调用process_sprite_group功能在rock_group里面
Phase two - 撞击

这部分,你要检测飞船和岩石的撞击。 发生撞击,岩石要毁坏并且玩家要丢失生命.  为了完成飞船岩石撞击:你要这么做:

  1. 在Sprite类里面添加一个collide功能。 这个应该有个other_object并返回True,如果有撞击返回False.  目前,这个其他对象将一直是你的飞船, 但我们也想要能用这个colide方法来侦测跟导弹的撞击。  可以使用这两个对象的半径侦测撞击。这需要你完成 get_position和get_radius在Sprite和get_radius.
  2. 完成group_collide函数。这个函数要有个set group和一个sprit other_object并经检查在other_object和group里的元素的撞击。如果有撞击, 撞击对象应该从group里面移除。为了避免移除你正在迭代的set(可能产生一个严重的bug), 通过 set(group)创建一个副本迭代. 这个函数应该返回 True或者False取决于是否有撞击。确定使用第一部分的collide方法 在sprites在group里面来完成这个任务。
  3. 在draw handler里面使用group_collide判定是否飞船撞击到岩石.要是这样,减少一条生命值。 现在你可以有负的生命值
现在你可以玩这个躲避岩石的游戏了。

Phase three - 导弹

在这部分你要产生一个导弹组,在按空格的时候产生新的导弹加入导弹组里面。需要如下步骤:

  1. 删掉a_missile并用missile_group代替。 复位导弹组为空组。修改my_ship里面的shoot函数来创造新导弹。 (一个Sprite类里面的实例) 并添加到missile_group里面. , 如果你用模板的代码,每次生产导弹的时候开火的声音应该自动播放
  2. 在draw handler里面,使用你的process_sprite_group函数来process  missile_group. 当你发射导弹的时候,你会注意到他们永远在飞。 为了修正这个,我们需要修改Sprite类和process_sprite_group.
  3. 在Sprite类里面的update函数。每次调用 update,sprite要增加。如果寿命大于或者等于sprite的生命,我们要删掉它。. 所以return返回False (意味着我们要保留它) 如果age少于寿命并且True(意味着我们要删掉它) 否则.
  4. 修改process_sprite_group来检查返回的值对sprites. 如果返回 True, 从group里面删掉sprite。再一次 你将希望迭代sprite群的副本在process_sprite_group.来避免删掉从相同的set在你迭代的。
Phase four - Collisions revisited  现在当导弹撞到岩石的时候要毁掉岩石。我们不能用group_collide,  因为我要检查两个group的碰撞。我们要做的就是加一个新的函数 :
  1.  完成最后一个函数group_group_collide ,啥呢,将两个grops的对象作为输入.   group_group_collide应该使用一个for循环在第一组副本元素迭代,然后第二组的所有元素调用group_collide.group_group_collide应该返回第一组元素跟第二在元素撞击并删除第一组中的这些元素。 你可能会发现set的discard 功能在这里很有用。
  2. 在draw handler里面调用 group_group_collide来检测导弹/岩石的碰撞。 根据导弹撞击的数目增加分数。
Phase five - Finish it off完成它

基本完了,再添加一点

  1. 添加代码到draw handler里面 , 要是生命数变成0了,游戏重置,并且出现启动画面.尤其, set the 设置startedFalse, 毁掉所有岩石并阻止任何岩石生成,直到游戏重新开始。
  2. 当游戏开始/重新开始. 确保生命数和分数也重置。开始生成岩石 .播放/重新开始加载在程序模板的soundtrack变量里面背景音乐
  3. 当你生成岩石的时候,你想让他们跟你的飞船有一定距离,否则岩石在你头上生成的时候你就挂了。.那就不好玩了.一个简单的方法来实现这个效果是这样的:如果生的岩石跟飞船太近,忽视岩石生产事件。
  4. 试着根据分数变化岩石的速度来将游戏难度提高
  5. 调整一些参数,以使游戏按我们的想法运行

额外的

 

  1. ,在Sprite类的draw功能里面,检查 self.animated是否是True .如果是的话根据age选择正确的碎片.图片是平铺的.如果self.animated是 False,应该像以前一样画sprite.
  2. 创建一个 explosion_group,全局变量并复位它成为一个空集合。
  3. 在group_collide里面,如果有碰撞,创建一个新的爆炸(一个Sprite类的实例)并添加到explosion_group里面,确认每个爆炸都播放爆炸声音。
  4. 在draw handler里面使用process_sprite_group表达explosion_group.

Grading rubric - 13 pts (scaled to 100 pts)


  • 1 pt - 程序产生很多岩石。
  • 1 pt - 程序正确的判定飞船是否跟岩石相撞
  • 1 pt - 飞船与岩石相撞,该岩石会消失
  • 1 pt - 飞船撞上岩石生命丢失一条
  • 1 pt - 程序产生多重导弹
  • 1 pt - 发射导弹的时候播放导弹发射声音
  • 1 pt - 如果导弹没有击中岩石,导弹在固定的时间后会消失
  • 1 pt - 程序可以正确的判定导弹和岩石是否碰撞
  • 1 pt - 相互碰撞的导弹和岩石会消失
  • 1 pt - 导弹岩石相撞分数要更新
  • 1 pt - 当生命数变成0,显示启动画面并且所有的岩石要消失
  • 1 pt - 当点击启动画面,生命重置为3,分数重置为0,并启动背景音乐。
  • 1 pt - 游戏只有在启动画面不存在还有游戏正在进行的时候产生岩石。
# implementation of Spaceship - program template for RiceRocksimport simpleguiimport mathimport random# globals for user interfaceWIDTH = 800HEIGHT = 600score = 0lives = 3time = 0started = Falsezjcs = 0class ImageInfo:    def __init__(self, center, size, radius = 0, lifespan = None, animated = False):        self.center = center        self.size = size        self.radius = radius        if lifespan:            self.lifespan = lifespan        else:            self.lifespan = float(‘inf‘)        self.animated = animated    def get_center(self):        return self.center    def get_size(self):        return self.size    def get_radius(self):        return self.radius    def get_lifespan(self):        return self.lifespan    def get_animated(self):        return self.animated    # art assets created by Kim Lathrop, may be freely re-used in non-commercial projects, please credit Kim    # debris images - debris1_brown.png, debris2_brown.png, debris3_brown.png, debris4_brown.png#                 debris1_blue.png, debris2_blue.png, debris3_blue.png, debris4_blue.png, debris_blend.pngdebris_info = ImageInfo([320, 240], [640, 480])debris_image = simplegui.load_image("http://commondatastorage.googleapis.com/codeskulptor-assets/lathrop/debris2_blue.png")# nebula images - nebula_brown.png, nebula_blue.pngnebula_info = ImageInfo([400, 300], [800, 600])nebula_image = simplegui.load_image("http://commondatastorage.googleapis.com/codeskulptor-assets/lathrop/nebula_blue.f2014.png")# splash imagesplash_info = ImageInfo([200, 150], [400, 300])splash_image = simplegui.load_image("http://commondatastorage.googleapis.com/codeskulptor-assets/lathrop/splash.png")# ship imageship_info = ImageInfo([45, 45], [90, 90], 35)ship_image = simplegui.load_image("http://commondatastorage.googleapis.com/codeskulptor-assets/lathrop/double_ship.png")# missile image - shot1.png, shot2.png, shot3.pngmissile_info = ImageInfo([5,5], [10, 10], 3, 50)missile_image = simplegui.load_image("http://commondatastorage.googleapis.com/codeskulptor-assets/lathrop/shot2.png")# asteroid images - asteroid_blue.png, asteroid_brown.png, asteroid_blend.pngasteroid_info = ImageInfo([45, 45], [90, 90], 40)asteroid_image = simplegui.load_image("http://commondatastorage.googleapis.com/codeskulptor-assets/lathrop/asteroid_blue.png")# animated explosion - explosion_orange.png, explosion_blue.png, explosion_blue2.png, explosion_alpha.pngexplosion_info = ImageInfo([64, 64], [128, 128], 17, 24, True)explosion_image = simplegui.load_image("http://commondatastorage.googleapis.com/codeskulptor-assets/lathrop/explosion_alpha.png")# sound assets purchased from sounddogs.com, please do not redistribute# .ogg versions of sounds are also available, just replace .mp3 by .oggsoundtrack = simplegui.load_sound("http://commondatastorage.googleapis.com/codeskulptor-assets/sounddogs/soundtrack.mp3")missile_sound = simplegui.load_sound("http://commondatastorage.googleapis.com/codeskulptor-assets/sounddogs/missile.mp3")missile_sound.set_volume(.5)ship_thrust_sound = simplegui.load_sound("http://commondatastorage.googleapis.com/codeskulptor-assets/sounddogs/thrust.mp3")explosion_sound = simplegui.load_sound("http://commondatastorage.googleapis.com/codeskulptor-assets/sounddogs/explosion.mp3")# helper functions to handle transformationsdef angle_to_vector(ang):    return [math.cos(ang), math.sin(ang)]def dist(p, q):    return math.sqrt((p[0] - q[0]) ** 2 + (p[1] - q[1]) ** 2)# Ship classclass Ship:    def __init__(self, pos, vel, angle, image, info):        self.pos = [pos[0], pos[1]]        self.vel = [vel[0], vel[1]]        self.thrust = False        self.angle = angle        self.angle_vel = 0        self.image = image        self.image_center = info.get_center()        self.image_size = info.get_size()        self.radius = info.get_radius()            def draw(self,canvas):        if self.thrust:            canvas.draw_image(self.image, [self.image_center[0] + self.image_size[0], self.image_center[1]] , self.image_size,                              self.pos, self.image_size, self.angle)        else:            canvas.draw_image(self.image, self.image_center, self.image_size,                              self.pos, self.image_size, self.angle)        # canvas.draw_circle(self.pos, self.radius, 1, "White", "White")    def update(self):        # update angle        self.angle += self.angle_vel                # update position        self.pos[0] = (self.pos[0] + self.vel[0]) % WIDTH        self.pos[1] = (self.pos[1] + self.vel[1]) % HEIGHT        # update velocity        if self.thrust:            acc = angle_to_vector(self.angle)            self.vel[0] += acc[0] * .1            self.vel[1] += acc[1] * .1                    self.vel[0] *= .99        self.vel[1] *= .99    def set_thrust(self, on):        self.thrust = on        if on:            ship_thrust_sound.rewind()            ship_thrust_sound.play()        else:            ship_thrust_sound.pause()           def increment_angle_vel(self):        self.angle_vel += .05            def decrement_angle_vel(self):        self.angle_vel -= .05            def shoot(self):        global a_missile, missile_group        forward = angle_to_vector(self.angle)        missile_pos = [self.pos[0] + self.radius * forward[0], self.pos[1] + self.radius * forward[1]]        missile_vel = [self.vel[0] + 6 * forward[0], self.vel[1] + 6 * forward[1]]        a_missile = Sprite(missile_pos, missile_vel, self.angle, 0, missile_image, missile_info, missile_sound)        missile_group.add(a_missile)    def get_position(self):        return self.pos      def get_radius(self):        return self.radius        # Sprite classclass Sprite:    def __init__(self, pos, vel, ang, ang_vel, image, info, sound = None):        self.pos = [pos[0],pos[1]]        self.vel = [vel[0],vel[1]]        self.angle = ang        self.angle_vel = ang_vel        self.image = image        self.image_center = info.get_center()        self.image_size = info.get_size()        self.radius = info.get_radius()        self.lifespan = info.get_lifespan()        self.animated = info.get_animated()        self.age = 0        if sound:            sound.rewind()            sound.play()       def draw(self, canvas):        canvas.draw_image(self.image, self.image_center, self.image_size,                          self.pos, self.image_size, self.angle)    def update(self):        # update angle        self.angle += self.angle_vel                # update position        self.pos[0] = (self.pos[0] + self.vel[0]) % WIDTH        self.pos[1] = (self.pos[1] + self.vel[1]) % HEIGHT        self.age += 1        if self.age < self.lifespan:            return False        else:            return True             def get_position(self):        return self.pos      def get_radius(self):        return self.radius            def collide(self, other_object):        if dist(self.pos, other_object.get_position()) <= self.radius + other_object.get_radius():            return True        else:            return False  #参考https://class.coursera.org/interactivepython-005/forum/thread?thread_id=5328def group_collide(group, other_object):    global collided    collided = False    group_copy = list(group)    for item in group_copy:        if item.collide(other_object) == True:            collided = True            group.discard(item)            return collided    def group_group_collide(group1, group2):    global zjcs    group1_copy = list(group1)    for i in group1_copy:        if group_collide(group2, i) == True:            group1.discard(i)            zjcs += 1            return True    # key handlers to control ship   def keydown(key):    if key == simplegui.KEY_MAP[‘left‘]:        my_ship.decrement_angle_vel()    elif key == simplegui.KEY_MAP[‘right‘]:        my_ship.increment_angle_vel()    elif key == simplegui.KEY_MAP[‘up‘]:        my_ship.set_thrust(True)    elif key == simplegui.KEY_MAP[‘space‘]:        my_ship.shoot()        def keyup(key):    if key == simplegui.KEY_MAP[‘left‘]:        my_ship.increment_angle_vel()    elif key == simplegui.KEY_MAP[‘right‘]:        my_ship.decrement_angle_vel()    elif key == simplegui.KEY_MAP[‘up‘]:        my_ship.set_thrust(False)            # mouseclick handlers that reset UI and conditions whether splash image is drawndef click(pos):    timer.start()    global started    center = [WIDTH / 2, HEIGHT / 2]    size = splash_info.get_size()    inwidth = (center[0] - size[0] / 2) < pos[0] < (center[0] + size[0] / 2)    inheight = (center[1] - size[1] / 2) < pos[1] < (center[1] + size[1] / 2)    if (not started) and inwidth and inheight:        started = True        soundtrack.rewind()        soundtrack.play()    def draw(canvas):    global time, started, lives, score, zjcs, rock_group        # animiate background    time += 1    wtime = (time / 4) % WIDTH    center = debris_info.get_center()    size = debris_info.get_size()    canvas.draw_image(nebula_image, nebula_info.get_center(), nebula_info.get_size(), [WIDTH / 2, HEIGHT / 2], [WIDTH, HEIGHT])    canvas.draw_image(debris_image, center, size, (wtime - WIDTH / 2, HEIGHT / 2), (WIDTH, HEIGHT))    canvas.draw_image(debris_image, center, size, (wtime + WIDTH / 2, HEIGHT / 2), (WIDTH, HEIGHT))    # draw UI    canvas.draw_text("Lives", [50, 50], 22, "White")    canvas.draw_text("Score", [680, 50], 22, "White")    if group_collide(rock_group, my_ship) == True:        lives -= 1        if lives < 1:            started = False            lives = 3            score = 0            zjcs = 0            rock_group = set([])            timer.stop()        #number -= 1    #canvas.draw_text(str(lives), [50, 80], 22, "White")    #canvas.draw_text(str(score), [680, 80], 22, "White")    if group_group_collide(missile_group, rock_group) == True:        score = zjcs * 10    # draw ship and sprites    my_ship.draw(canvas)    #a_rock.draw(canvas)    process_sprite_group(canvas,rock_group)    process_sprite_group(canvas,missile_group)    #for n in rock_group:     #   n.draw(canvas)    #a_missile.draw(canvas)注释掉这一行,导弹就不是一直存在了        # update ship and sprites    my_ship.update()    #a_rock.update()?????????????    #a_missile.update()    # draw splash screen if not started    if not started:        canvas.draw_image(splash_image, splash_info.get_center(),                           splash_info.get_size(), [WIDTH / 2, HEIGHT / 2],                           splash_info.get_size())    canvas.draw_text("Lives", [50, 50], 22, "White")    canvas.draw_text("Score", [680, 50], 22, "White")            canvas.draw_text(str(lives), [50, 80], 22, "White")#放下面,可以一直在上面显示    canvas.draw_text(str(score), [680, 80], 22, "White")# timer handler that spawns a rock    def rock_spawner():    global a_rock, rock_group, zjcs    rock_pos = [random.randrange(0, WIDTH), random.randrange(0, HEIGHT)]        rock_vel = [random.random() * .6 - .3, random.random() * .6 - .3]    if zjcs > 5:#得分超过5,速度加快        rock_vel = [random.random() * 6 - .3, random.random() * 6 - .3]    rock_avel = random.random() * .2 - .1    a_rock = Sprite(rock_pos, rock_vel, 0, rock_avel, asteroid_image, asteroid_info)    #rock_group.add(a_rock)    if len(rock_group) < 13 and dist(rock_pos, my_ship.pos) > 150:        rock_group.add(a_rock)            def process_sprite_group(canvas, group):    #if a_missile.update == True:        #missile_group.remove(a_missile)    for m in list(group):        #m.update()        if m.update() == True:            missile_group.remove(m)        m.draw(canvas)                            # initialize stuffframe = simplegui.create_frame("Asteroids", WIDTH, HEIGHT)# initialize ship and two spritesmy_ship = Ship([WIDTH / 2, HEIGHT / 2], [0, 0], 0, ship_image, ship_info)a_rock = Sprite([WIDTH / 3, HEIGHT / 3], [1, 1], 0, .1, asteroid_image, asteroid_info)rock_group = set([])    a_missile = Sprite([2 * WIDTH / 3, 2 * HEIGHT / 3], [-1,1], 0, 0, missile_image, missile_info, missile_sound)missile_group = set([])# register handlersframe.set_keyup_handler(keyup)frame.set_keydown_handler(keydown)frame.set_mouseclick_handler(click)frame.set_draw_handler(draw)timer = simplegui.create_timer(1000.0, rock_spawner)# get things rollingtimer.stop()frame.start()

  

Rice Rock