forked from len0rd/rockbox
		
	Plugins on the ZEN/ZEN X-Fi require to increase the plugin buffer size. Change-Id: If4651c87b402060faa24530985c6e871379c8ea1
		
			
				
	
	
		
			1893 lines
		
	
	
	
		
			54 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			1893 lines
		
	
	
	
		
			54 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /***************************************************************************
 | |
|  *             __________               __   ___.
 | |
|  *   Open      \______   \ ____   ____ |  | _\_ |__   _______  ___
 | |
|  *   Source     |       _//  _ \_/ ___\|  |/ /| __ \ /  _ \  \/  /
 | |
|  *   Jukebox    |    |   (  <_> )  \___|    < | \_\ (  <_> > <  <
 | |
|  *   Firmware   |____|_  /\____/ \___  >__|_ \|___  /\____/__/\_ \
 | |
|  *                     \/            \/     \/    \/            \/
 | |
|  * $Id$
 | |
|  *
 | |
|  * Copyright (C) 2006 Albert Veli
 | |
|  *
 | |
|  * 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 2
 | |
|  * of the License, or (at your option) any later version.
 | |
|  *
 | |
|  * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
 | |
|  * KIND, either express or implied.
 | |
|  *
 | |
|  ****************************************************************************/
 | |
| 
 | |
| /* Improvised creds goes to:
 | |
|  *
 | |
|  * - Anders Clausen for ingeniously inventing the name Invadrox.
 | |
|  * - Linus Nielsen-Feltzing for patiently answering n00b questions.
 | |
|  */
 | |
| 
 | |
| #include "plugin.h"
 | |
| #include "lib/highscore.h"
 | |
| #include "lib/helper.h"
 | |
| 
 | |
| /* bitmaps */
 | |
| #include "pluginbitmaps/invadrox_background.h"
 | |
| 
 | |
| /* get dimensions for later use from the bitmaps */
 | |
| #include "pluginbitmaps/invadrox_aliens.h"
 | |
| #include "pluginbitmaps/invadrox_ships.h"
 | |
| #include "pluginbitmaps/invadrox_bombs.h"
 | |
| #include "pluginbitmaps/invadrox_alien_explode.h"
 | |
| #include "pluginbitmaps/invadrox_shield.h"
 | |
| #include "pluginbitmaps/invadrox_ufo.h"
 | |
| #include "pluginbitmaps/invadrox_ufo_explode.h"
 | |
| #include "pluginbitmaps/invadrox_numbers.h"
 | |
| #include "pluginbitmaps/invadrox_fire.h"
 | |
| #define ALIEN_WIDTH           (BMPWIDTH_invadrox_aliens/2)
 | |
| #define ALIEN_HEIGHT          (BMPHEIGHT_invadrox_aliens/3)
 | |
| #define SHIP_WIDTH            BMPWIDTH_invadrox_ships
 | |
| #define SHIP_HEIGHT           (BMPHEIGHT_invadrox_ships/3)
 | |
| #define BOMB_WIDTH            (BMPWIDTH_invadrox_bombs/3)
 | |
| #define BOMB_HEIGHT           (BMPHEIGHT_invadrox_bombs/6)
 | |
| #define ALIEN_EXPLODE_WIDTH   BMPWIDTH_invadrox_alien_explode
 | |
| #define ALIEN_EXPLODE_HEIGHT  BMPHEIGHT_invadrox_alien_explode
 | |
| #define SHIELD_WIDTH          BMPWIDTH_invadrox_shield
 | |
| #define SHIELD_HEIGHT         BMPHEIGHT_invadrox_shield
 | |
| #define UFO_WIDTH             BMPWIDTH_invadrox_ufo
 | |
| #define UFO_HEIGHT            BMPHEIGHT_invadrox_ufo
 | |
| #define UFO_EXPLODE_WIDTH     BMPWIDTH_invadrox_ufo_explode
 | |
| #define UFO_EXPLODE_HEIGHT    BMPHEIGHT_invadrox_ufo_explode
 | |
| #define NUMBERS_WIDTH         (BMPWIDTH_invadrox_numbers/10)
 | |
| #define FONT_HEIGHT           BMPHEIGHT_invadrox_numbers
 | |
| #define FIRE_WIDTH            BMPWIDTH_invadrox_fire
 | |
| #define FIRE_HEIGHT           BMPHEIGHT_invadrox_fire
 | |
| 
 | |
| 
 | |
| 
 | |
| /* Original graphics is only 1bpp so it should be portable
 | |
|  * to most targets. But for now, only support the simple ones.
 | |
|  */
 | |
| #ifndef HAVE_LCD_BITMAP
 | |
|     #error INVADROX: Unsupported LCD
 | |
| #endif
 | |
| 
 | |
| #if (LCD_DEPTH < 2)
 | |
|     #error INVADROX: Unsupported LCD
 | |
| #endif
 | |
| 
 | |
| /* #define DEBUG */
 | |
| #ifdef DEBUG
 | |
| #define DBG(format, arg...) { DEBUGF("%s: " format, __FUNCTION__, ## arg); }
 | |
| #else
 | |
| #define DBG(format, arg...) {}
 | |
| #endif
 | |
| 
 | |
| #ifndef ABS
 | |
| #define ABS(a) (((a) < 0) ? -(a) : (a))
 | |
| #endif
 | |
| 
 | |
| #if CONFIG_KEYPAD == IRIVER_H100_PAD
 | |
| 
 | |
| #define QUIT BUTTON_OFF
 | |
| #define LEFT BUTTON_LEFT
 | |
| #define RIGHT BUTTON_RIGHT
 | |
| #define FIRE BUTTON_ON
 | |
| 
 | |
| #elif CONFIG_KEYPAD == IRIVER_H300_PAD
 | |
| 
 | |
| #define QUIT BUTTON_OFF
 | |
| #define LEFT BUTTON_LEFT
 | |
| #define RIGHT BUTTON_RIGHT
 | |
| #define FIRE BUTTON_SELECT
 | |
| 
 | |
| #elif (CONFIG_KEYPAD == IRIVER_H10_PAD)
 | |
| 
 | |
| #define QUIT BUTTON_POWER
 | |
| #define LEFT BUTTON_LEFT
 | |
| #define RIGHT BUTTON_RIGHT
 | |
| #define FIRE BUTTON_PLAY
 | |
| 
 | |
| #elif (CONFIG_KEYPAD == IPOD_4G_PAD) || \
 | |
|       (CONFIG_KEYPAD == IPOD_3G_PAD) || \
 | |
|       (CONFIG_KEYPAD == IPOD_1G2G_PAD)
 | |
| 
 | |
| #define QUIT BUTTON_MENU
 | |
| #define LEFT BUTTON_LEFT
 | |
| #define RIGHT BUTTON_RIGHT
 | |
| #define FIRE BUTTON_SELECT
 | |
| 
 | |
| #elif CONFIG_KEYPAD == IAUDIO_X5M5_PAD
 | |
| 
 | |
| #define QUIT BUTTON_POWER
 | |
| #define LEFT BUTTON_LEFT
 | |
| #define RIGHT BUTTON_RIGHT
 | |
| #define FIRE BUTTON_SELECT
 | |
| 
 | |
| #elif CONFIG_KEYPAD == GIGABEAT_PAD \
 | |
|    || CONFIG_KEYPAD == SAMSUNG_YPR0_PAD
 | |
| 
 | |
| #define QUIT BUTTON_POWER
 | |
| #define LEFT BUTTON_LEFT
 | |
| #define RIGHT BUTTON_RIGHT
 | |
| #define FIRE BUTTON_SELECT
 | |
| 
 | |
| #elif (CONFIG_KEYPAD == SANSA_E200_PAD) || \
 | |
|       (CONFIG_KEYPAD == SANSA_CONNECT_PAD)
 | |
| 
 | |
| #define QUIT BUTTON_POWER
 | |
| #define LEFT BUTTON_LEFT
 | |
| #define RIGHT BUTTON_RIGHT
 | |
| #define FIRE BUTTON_SELECT
 | |
| 
 | |
| #elif CONFIG_KEYPAD == SANSA_FUZE_PAD
 | |
| 
 | |
| #define QUIT (BUTTON_HOME|BUTTON_REPEAT)
 | |
| #define LEFT BUTTON_LEFT
 | |
| #define RIGHT BUTTON_RIGHT
 | |
| #define FIRE BUTTON_SELECT
 | |
| 
 | |
| #elif CONFIG_KEYPAD == TATUNG_TPJ1022_PAD
 | |
| 
 | |
| /* TODO: Figure out which buttons to use for Tatung Elio TPJ-1022 */
 | |
| #define QUIT BUTTON_AB
 | |
| #define LEFT BUTTON_LEFT
 | |
| #define RIGHT BUTTON_RIGHT
 | |
| #define FIRE BUTTON_MENU
 | |
| 
 | |
| #elif CONFIG_KEYPAD == GIGABEAT_S_PAD
 | |
| 
 | |
| #define QUIT BUTTON_BACK
 | |
| #define LEFT BUTTON_LEFT
 | |
| #define RIGHT BUTTON_RIGHT
 | |
| #define FIRE BUTTON_SELECT
 | |
| 
 | |
| #elif CONFIG_KEYPAD == COWON_D2_PAD
 | |
| 
 | |
| #define QUIT BUTTON_POWER
 | |
| #define LEFT BUTTON_MINUS
 | |
| #define RIGHT BUTTON_PLUS
 | |
| #define FIRE BUTTON_MENU
 | |
| 
 | |
| #elif CONFIG_KEYPAD == IAUDIO67_PAD
 | |
| 
 | |
| #define QUIT BUTTON_POWER
 | |
| #define LEFT BUTTON_LEFT
 | |
| #define RIGHT BUTTON_RIGHT
 | |
| #define FIRE BUTTON_PLAY
 | |
| 
 | |
| #elif CONFIG_KEYPAD == CREATIVEZVM_PAD
 | |
| 
 | |
| #define QUIT BUTTON_BACK
 | |
| #define LEFT BUTTON_LEFT
 | |
| #define RIGHT BUTTON_RIGHT
 | |
| #define FIRE BUTTON_SELECT
 | |
| 
 | |
| #elif CONFIG_KEYPAD == CREATIVE_ZENXFI3_PAD
 | |
| 
 | |
| #define QUIT BUTTON_POWER
 | |
| #define LEFT BUTTON_BACK
 | |
| #define RIGHT BUTTON_MENU
 | |
| #define FIRE BUTTON_PLAY
 | |
| 
 | |
| #elif CONFIG_KEYPAD == PHILIPS_HDD6330_PAD
 | |
| 
 | |
| #define QUIT BUTTON_POWER
 | |
| #define LEFT BUTTON_LEFT
 | |
| #define RIGHT BUTTON_RIGHT
 | |
| #define FIRE BUTTON_PLAY
 | |
| 
 | |
| #elif CONFIG_KEYPAD == PHILIPS_SA9200_PAD
 | |
| 
 | |
| #define QUIT BUTTON_POWER
 | |
| #define LEFT BUTTON_PREV
 | |
| #define RIGHT BUTTON_NEXT
 | |
| #define FIRE BUTTON_PLAY
 | |
| 
 | |
| #elif CONFIG_KEYPAD == ONDAVX747_PAD || \
 | |
| CONFIG_KEYPAD == ONDAVX777_PAD || \
 | |
| CONFIG_KEYPAD == MROBE500_PAD
 | |
| 
 | |
| #define QUIT BUTTON_POWER
 | |
| 
 | |
| #elif (CONFIG_KEYPAD == SAMSUNG_YH820_PAD) || \
 | |
|       (CONFIG_KEYPAD == SAMSUNG_YH920_PAD)
 | |
| 
 | |
| #define QUIT  BUTTON_REC
 | |
| #define LEFT  BUTTON_LEFT
 | |
| #define RIGHT BUTTON_RIGHT
 | |
| #define FIRE  BUTTON_PLAY
 | |
| 
 | |
| #elif CONFIG_KEYPAD == PBELL_VIBE500_PAD
 | |
| 
 | |
| #define QUIT  BUTTON_REC
 | |
| #define LEFT  BUTTON_PREV
 | |
| #define RIGHT BUTTON_NEXT
 | |
| #define FIRE  BUTTON_OK
 | |
| 
 | |
| #elif CONFIG_KEYPAD == MPIO_HD300_PAD
 | |
| 
 | |
| #define QUIT  BUTTON_MENU
 | |
| #define LEFT  BUTTON_REW
 | |
| #define RIGHT BUTTON_FF
 | |
| #define FIRE  BUTTON_ENTER
 | |
| 
 | |
| #elif CONFIG_KEYPAD == SANSA_FUZEPLUS_PAD
 | |
| 
 | |
| #define QUIT  BUTTON_POWER
 | |
| #define LEFT  BUTTON_LEFT
 | |
| #define RIGHT BUTTON_RIGHT
 | |
| #define FIRE  BUTTON_SELECT
 | |
| 
 | |
| #elif CONFIG_KEYPAD == SONY_NWZ_PAD
 | |
| 
 | |
| #define QUIT  BUTTON_BACK
 | |
| #define LEFT  BUTTON_LEFT
 | |
| #define RIGHT BUTTON_RIGHT
 | |
| #define FIRE  BUTTON_PLAY
 | |
| 
 | |
| #elif CONFIG_KEYPAD == CREATIVE_ZEN_PAD
 | |
| 
 | |
| #define QUIT  BUTTON_BACK
 | |
| #define LEFT  BUTTON_LEFT
 | |
| #define RIGHT BUTTON_RIGHT
 | |
| #define FIRE  BUTTON_SELECT
 | |
| 
 | |
| #elif (CONFIG_KEYPAD == HM60X_PAD) || \
 | |
|     (CONFIG_KEYPAD == HM801_PAD)
 | |
| 
 | |
| #define QUIT  BUTTON_POWER
 | |
| #define LEFT  BUTTON_LEFT
 | |
| #define RIGHT BUTTON_RIGHT
 | |
| #define FIRE  BUTTON_SELECT
 | |
| 
 | |
| #else
 | |
|     #error INVADROX: Unsupported keypad
 | |
| #endif
 | |
| 
 | |
| #ifndef RC_QUIT
 | |
| #define RC_QUIT 0
 | |
| #endif
 | |
| 
 | |
| #ifdef HAVE_TOUCHSCREEN
 | |
| 
 | |
| #ifndef QUIT
 | |
| #define QUIT 0
 | |
| #endif
 | |
| #ifndef LEFT
 | |
| #define LEFT 0
 | |
| #endif
 | |
| #ifndef RIGHT
 | |
| #define RIGHT 0
 | |
| #endif
 | |
| #ifndef FIRE
 | |
| #define FIRE 0
 | |
| #endif
 | |
| 
 | |
| #define TOUCHSCREEN_QUIT    BUTTON_TOPLEFT
 | |
| #define TOUCHSCREEN_LEFT    (BUTTON_MIDLEFT | BUTTON_BOTTOMLEFT)
 | |
| #define TOUCHSCREEN_RIGHT   (BUTTON_MIDRIGHT | BUTTON_BOTTOMRIGHT)
 | |
| #define TOUCHSCREEN_FIRE    (BUTTON_CENTER | BUTTON_BOTTOMMIDDLE)
 | |
| 
 | |
| #define ACTION_QUIT     (QUIT | TOUCHSCREEN_QUIT | RC_QUIT)
 | |
| #define ACTION_LEFT     (LEFT | TOUCHSCREEN_LEFT)
 | |
| #define ACTION_RIGHT    (RIGHT | TOUCHSCREEN_RIGHT)
 | |
| #define ACTION_FIRE     (FIRE | TOUCHSCREEN_FIRE)
 | |
| 
 | |
| #else /* HAVE_TOUCHSCREEN */
 | |
| 
 | |
| #define ACTION_QUIT  (QUIT | RC_QUIT)
 | |
| #define ACTION_LEFT  LEFT
 | |
| #define ACTION_RIGHT RIGHT
 | |
| #define ACTION_FIRE  FIRE
 | |
| 
 | |
| #endif
 | |
| 
 | |
| #ifndef UNUSED
 | |
| #define UNUSED __attribute__ ((unused))
 | |
| #endif
 | |
| 
 | |
| /* Defines common to all models */
 | |
| #define UFO_Y (SCORENUM_Y + FONT_HEIGHT + ALIEN_HEIGHT)
 | |
| #define PLAYFIELD_Y (LCD_HEIGHT - SHIP_HEIGHT - 2)
 | |
| #define PLAYFIELD_WIDTH (LCD_WIDTH - 2 * PLAYFIELD_X)
 | |
| #define LEVEL_X (LCD_WIDTH - PLAYFIELD_X - LIVES_X - 2 * NUMBERS_WIDTH - 3 * NUM_SPACING)
 | |
| #define SHIP_MIN_X (PLAYFIELD_X + PLAYFIELD_WIDTH / 5 - SHIELD_WIDTH / 2 - SHIP_WIDTH)
 | |
| #define SHIP_MAX_X (PLAYFIELD_X + 4 * PLAYFIELD_WIDTH / 5 + SHIELD_WIDTH / 2)
 | |
| /* SCORE_Y = 0 for most targets. Gigabeat redefines it later. */
 | |
| #define SCORE_Y 0
 | |
| #define MAX_LIVES 8
 | |
| 
 | |
| 
 | |
| /* m:robe 500 defines */
 | |
| #if ((LCD_WIDTH == 640) && (LCD_HEIGHT == 480)) || \
 | |
|     ((LCD_WIDTH == 480) && (LCD_HEIGHT == 640))
 | |
| 
 | |
| /* Original arcade game size 224x240, 1bpp with
 | |
|  * red overlay at top and green overlay at bottom.
 | |
|  *
 | |
|  * M:Robe 500: 640x480x16
 | |
|  * ======================
 | |
|  */
 | |
| 
 | |
| #define ARCADISH_GRAPHICS
 | |
| #define PLAYFIELD_X 48
 | |
| #define SHIP_Y (PLAYFIELD_Y - 2 * SHIP_HEIGHT)
 | |
| #define ALIEN_START_Y (UFO_Y + ALIEN_HEIGHT)
 | |
| #define SCORENUM_X (PLAYFIELD_X + NUMBERS_WIDTH)
 | |
| #define SCORENUM_Y (SCORE_Y + FONT_HEIGHT + 2)
 | |
| #define HISCORENUM_X (LCD_WIDTH - PLAYFIELD_X - 1 - 6 * NUMBERS_WIDTH - 5 * NUM_SPACING)
 | |
| #define SHIELD_Y (PLAYFIELD_Y - 5 * SHIP_HEIGHT)
 | |
| #define LIVES_X 10
 | |
| #define MAX_Y 18
 | |
| 
 | |
| /* iPod Video defines */
 | |
| #elif (LCD_WIDTH == 320) && (LCD_HEIGHT == 240)
 | |
| 
 | |
| /* Original arcade game size 224x240, 1bpp with
 | |
|  * red overlay at top and green overlay at bottom.
 | |
|  *
 | |
|  * iPod Video: 320x240x16
 | |
|  * ======================
 | |
|  * X: 48p padding at left/right gives 224p playfield in middle.
 | |
|  *    10p "border" gives 204p actual playfield. UFO use full 224p.
 | |
|  * Y: Use full 240p.
 | |
|  *
 | |
|  * MAX_X = (204 - 12) / 2 - 1 = 95
 | |
|  *
 | |
|  * Y: Score text 7       0
 | |
|  *    Space      10      7
 | |
|  *    Score      7       17
 | |
|  *    Space      8       24
 | |
|  * 3  Ufo        7       32
 | |
|  * 2  Space      Aliens start at 32 + 3 * 8 = 56
 | |
|  * 0  aliens     9*8     56  -
 | |
|  *    space     ~7*8     128  | 18.75 aliens space between
 | |
|  *    shield     2*8     182  | first alien and ship.
 | |
|  *    space      8       198  | MAX_Y = 18
 | |
|  *    ship       8       206 -
 | |
|  *    space      2*8     214
 | |
|  *    hline      1       230 - PLAYFIELD_Y
 | |
|  *    bottom border 10   240
 | |
|  *    Lives and Level goes inside bottom border
 | |
|  */
 | |
| 
 | |
| #define ARCADISH_GRAPHICS
 | |
| #define PLAYFIELD_X 48
 | |
| #define SHIP_Y (PLAYFIELD_Y - 3 * SHIP_HEIGHT)
 | |
| #define ALIEN_START_Y (UFO_Y + 3 * ALIEN_HEIGHT)
 | |
| #define SCORENUM_X (PLAYFIELD_X + NUMBERS_WIDTH)
 | |
| #define SCORENUM_Y SCORE_Y + (2 * (FONT_HEIGHT + 1) + 1)
 | |
| #define HISCORENUM_X (LCD_WIDTH - PLAYFIELD_X - 1 - 6 * NUMBERS_WIDTH - 5 * NUM_SPACING)
 | |
| #define SHIELD_Y (PLAYFIELD_Y - 6 * SHIP_HEIGHT)
 | |
| #define LIVES_X 10
 | |
| #define MAX_Y 18
 | |
| 
 | |
| #elif (LCD_WIDTH == 176) && (LCD_HEIGHT == 220)
 | |
| 
 | |
| /* Sandisk Sansa e200: 176x220x16
 | |
|  * ==============================
 | |
|  * X: No padding. 8p border -> 160p playfield.
 | |
|  *
 | |
|  * 8p Aliens with 3p spacing -> 88 + 30 = 118p aliens block.
 | |
|  * (160 - 118) / 2 = 21 rounds for whole block (more than original)
 | |
|  * MAX_X = (160 - 8) / 2 - 1 = 75 rounds for single alien (less than original)
 | |
|  *
 | |
|  *    LOGO       70       0
 | |
|  *    Score text 5        70
 | |
|  *    Space      5        75
 | |
|  * Y  Score      5        80
 | |
|  *    Space      10       85
 | |
|  * 2  Ufo        5        95
 | |
|  * 2  Space      10      100
 | |
|  * 0  aliens     9*5     110 -
 | |
|  *    space     ~7*5     155  | 18.6 aliens space between
 | |
|  *    shield     2*5     188  | first alien and ship.
 | |
|  *    space      5       198  | MAX_Y = 18
 | |
|  *    ship       5       203 -
 | |
|  *    space      5       208
 | |
|  *    hline      1       213 PLAYFIELD_Y
 | |
|  *    bottom border 6
 | |
|  *    LCD_HEIGHT         220
 | |
|  *    Lives and Level goes inside bottom border
 | |
|  */
 | |
| 
 | |
| #define SMALL_GRAPHICS
 | |
| #define PLAYFIELD_X 0
 | |
| #define SHIP_Y (PLAYFIELD_Y - 2 * SHIP_HEIGHT)
 | |
| #define SHIELD_Y (SHIP_Y - SHIP_HEIGHT - SHIELD_HEIGHT)
 | |
| #define ALIEN_START_Y (UFO_Y + 3 * SHIP_HEIGHT)
 | |
| /* Redefine SCORE_Y */
 | |
| #undef SCORE_Y
 | |
| #define SCORE_Y 70
 | |
| #define SCORENUM_X (PLAYFIELD_X + NUMBERS_WIDTH)
 | |
| #define SCORENUM_Y (SCORE_Y + 2 * FONT_HEIGHT)
 | |
| #define HISCORENUM_X (LCD_WIDTH - PLAYFIELD_X - 1 - 6 * NUMBERS_WIDTH - 5 * NUM_SPACING)
 | |
| #define LIVES_X 8
 | |
| #define MAX_Y 18
 | |
| 
 | |
| 
 | |
| #elif (LCD_WIDTH == 176) && (LCD_HEIGHT == 132)
 | |
| 
 | |
| /* iPod Nano: 176x132x16
 | |
|  * ======================
 | |
|  * X: No padding. 8p border -> 160p playfield.
 | |
|  *
 | |
|  *    LIVES_X 8
 | |
|  *    ALIEN_WIDTH 8
 | |
|  *    ALIEN_HEIGHT 5
 | |
|  *    ALIEN_SPACING 3
 | |
|  *    SHIP_WIDTH 10
 | |
|  *    SHIP_HEIGHT 5
 | |
|  *    FONT_HEIGHT 5
 | |
|  *    UFO_WIDTH 10
 | |
|  *    UFO_HEIGHT 5
 | |
|  *    SHIELD_WIDTH 15
 | |
|  *    SHIELD_HEIGHT 10
 | |
|  *    MAX_X 75
 | |
|  *    MAX_Y = 18
 | |
|  *    ALIEN_START_Y (UFO_Y + 12)
 | |
|  *
 | |
|  * 8p Aliens with 3p spacing -> 88 + 30 = 118p aliens block.
 | |
|  * (160 - 118) / 2 = 21 rounds for whole block (more than original)
 | |
|  * MAX_X = (160 - 8) / 2 - 1 = 75 rounds for single alien (less than original)
 | |
|  *
 | |
|  * Y: Scoreline  5         0 (combine scoretext and numbers on same line)
 | |
|  *    Space      5         5
 | |
|  * 1  Ufo        5        10
 | |
|  * 3  Space      7        15
 | |
|  * 2  aliens     9*5      22 -
 | |
|  *    space     ~7*5      67  | Just above 18 aliens space between
 | |
|  *    shield     2*5     100  | first alien and ship.
 | |
|  *    space      5       110  | MAX_Y = 18
 | |
|  *    ship       5       115 -
 | |
|  *    space      5       120
 | |
|  *    hline      1       125 PLAYFIELD_Y
 | |
|  *    bottom border 6    126
 | |
|  *    LCD_HEIGHT         131
 | |
|  *    Lives and Level goes inside bottom border
 | |
|  */
 | |
| 
 | |
| #define SMALL_GRAPHICS
 | |
| #define PLAYFIELD_X 0
 | |
| #define SHIP_Y (PLAYFIELD_Y - 2 * SHIP_HEIGHT)
 | |
| #define ALIEN_START_Y (UFO_Y + 12)
 | |
| #define SCORENUM_X (PLAYFIELD_X + 6 * NUMBERS_WIDTH + 5 * NUM_SPACING)
 | |
| #define SCORENUM_Y SCORE_Y
 | |
| #define HISCORENUM_X (LCD_WIDTH - PLAYFIELD_X - 4 * NUMBERS_WIDTH - 3 * NUM_SPACING)
 | |
| #define SHIELD_Y (SHIP_Y - SHIP_HEIGHT - SHIELD_HEIGHT)
 | |
| #define LIVES_X 8
 | |
| #define MAX_Y 18
 | |
| 
 | |
| #elif (LCD_WIDTH == 160) && (LCD_HEIGHT == 128)
 | |
| 
 | |
| /* iAudio X5, iRiver H10 20Gb, iPod 3g/4g, H100, M5: 160x128
 | |
|  * =========================================================
 | |
|  * X: No padding. No border -> 160p playfield.
 | |
|  *
 | |
|  *    LIVES_X 0
 | |
|  *    ALIEN_WIDTH 8
 | |
|  *    ALIEN_HEIGHT 5
 | |
|  *    ALIEN_SPACING 3
 | |
|  *    SHIP_WIDTH 10
 | |
|  *    SHIP_HEIGHT 5
 | |
|  *    FONT_HEIGHT 5
 | |
|  *    UFO_WIDTH 10
 | |
|  *    UFO_HEIGHT 5
 | |
|  *    SHIELD_WIDTH 15
 | |
|  *    SHIELD_HEIGHT 10
 | |
|  *    MAX_X 75
 | |
|  *    MAX_Y = 18
 | |
|  *    ALIEN_START_Y (UFO_Y + 10)
 | |
|  *
 | |
|  * 8p Aliens with 3p spacing -> 88 + 30 = 118p aliens block.
 | |
|  * (160 - 118) / 2 = 21 rounds for whole block (more than original)
 | |
|  * MAX_X = (160 - 8) / 2 - 1 = 75 rounds for single alien (less than original)
 | |
|  *
 | |
|  * Y: Scoreline  5         0 (combine scoretext and numbers on same line)
 | |
|  *    Space      5         5
 | |
|  * 1  Ufo        5        10
 | |
|  * 2  Space      5        15
 | |
|  * 8  aliens     9*5      20 -
 | |
|  *    space     ~6*5      65  | Just above 18 aliens space between
 | |
|  *    shield     2*5      96  | first alien and ship.
 | |
|  *    space      5       106  | MAX_Y = 18
 | |
|  *    ship       5       111 -
 | |
|  *    space      5       116
 | |
|  *    hline      1       121 PLAYFIELD_Y
 | |
|  *    bottom border 6    122
 | |
|  *    LCD_HEIGHT         128
 | |
|  *    Lives and Level goes inside bottom border
 | |
|  */
 | |
| 
 | |
| #define SMALL_GRAPHICS
 | |
| #define PLAYFIELD_X 0
 | |
| #define SHIP_Y (PLAYFIELD_Y - 2 * SHIP_HEIGHT)
 | |
| #define ALIEN_START_Y (UFO_Y + 10)
 | |
| #define SCORENUM_X (PLAYFIELD_X + 6 * NUMBERS_WIDTH + 5 * NUM_SPACING)
 | |
| #define SCORENUM_Y SCORE_Y
 | |
| #define HISCORENUM_X (LCD_WIDTH - PLAYFIELD_X - 4 * NUMBERS_WIDTH - 3 * NUM_SPACING)
 | |
| #define SHIELD_Y (SHIP_Y - SHIP_HEIGHT - SHIELD_HEIGHT)
 | |
| #define LIVES_X 0
 | |
| #define MAX_Y 18
 | |
| 
 | |
| 
 | |
| #elif (LCD_WIDTH == 240) && ((LCD_HEIGHT == 320) || (LCD_HEIGHT == 400))
 | |
| 
 | |
| /* Gigabeat: 240x320x16
 | |
|  * ======================
 | |
|  * X: 8p padding at left/right gives 224p playfield in middle.
 | |
|  *    10p "border" gives 204p actual playfield. UFO use full 224p.
 | |
|  * Y: Use bottom 240p for playfield and top 80 pixels for logo.
 | |
|  *
 | |
|  * MAX_X = (204 - 12) / 2 - 1 = 95
 | |
|  *
 | |
|  * Y: Score text 7       0   + 80
 | |
|  *    Space      10      7   + 80
 | |
|  *    Score      7       17  + 80
 | |
|  *    Space      8       24  + 80
 | |
|  * 3  Ufo        7       32  + 80
 | |
|  * 2  Space      Aliens start at 32 + 3 * 8 = 56
 | |
|  * 0  aliens     9*8     56  -
 | |
|  *    space     ~7*8     128  | 18.75 aliens space between
 | |
|  *    shield     2*8     182  | first alien and ship.
 | |
|  *    space      8       198  | MAX_Y = 18
 | |
|  *    ship       8       206 -
 | |
|  *    space      2*8     214
 | |
|  *    hline      1       230 310  - PLAYFIELD_Y
 | |
|  *    bottom border 10   240 320
 | |
|  *    Lives and Level goes inside bottom border
 | |
|  */
 | |
| 
 | |
| #define ARCADISH_GRAPHICS
 | |
| #define PLAYFIELD_X 8
 | |
| #define SHIP_Y (PLAYFIELD_Y - 3 * SHIP_HEIGHT)
 | |
| #define ALIEN_START_Y (UFO_Y + 3 * ALIEN_HEIGHT)
 | |
| /* Redefine SCORE_Y */
 | |
| #undef SCORE_Y
 | |
| #define SCORE_Y 80
 | |
| #define SCORENUM_X (PLAYFIELD_X + NUMBERS_WIDTH)
 | |
| #define SCORENUM_Y SCORE_Y + (2 * (FONT_HEIGHT + 1) + 1)
 | |
| #define HISCORENUM_X (LCD_WIDTH - PLAYFIELD_X - 1 - 6 * NUMBERS_WIDTH - 5 * NUM_SPACING)
 | |
| #define SHIELD_Y (PLAYFIELD_Y - 6 * SHIP_HEIGHT)
 | |
| #define LIVES_X 10
 | |
| #define MAX_Y 18
 | |
| 
 | |
| #elif (LCD_WIDTH == 220) && (LCD_HEIGHT == 176)
 | |
| 
 | |
| /* TPJ1022, H300, iPod Color: 220x176x16
 | |
|  * ============================
 | |
|  * X: 0p padding at left/right gives 220p playfield in middle.
 | |
|  *    8p "border" gives 204p actual playfield. UFO use full 220p.
 | |
|  * Y: Use full 176p for playfield.
 | |
|  *
 | |
|  * MAX_X = (204 - 12) / 2 - 1 = 95
 | |
|  *
 | |
|  * Y: Score text 7       0
 | |
|  *    Space      8       7
 | |
|  * 1  Ufo        7       15
 | |
|  * 7  Space      Aliens start at 15 + 3 * 8 = 56
 | |
|  * 6  aliens     9*8     25  -
 | |
|  *    space     ~7*8     103  | 15.6 aliens space between
 | |
|  *    shield     2*8     126  | first alien and ship.
 | |
|  *    space      8       142  | MAX_Y = 15
 | |
|  *    ship       8       150 -
 | |
|  *    space      8       158
 | |
|  *    hline      1       166 - PLAYFIELD_Y
 | |
|  *    bottom border 10   176
 | |
|  *    Lives and Level goes inside bottom border
 | |
|  */
 | |
| 
 | |
| #define ARCADISH_GRAPHICS
 | |
| #define PLAYFIELD_X 0
 | |
| #define SHIP_Y (PLAYFIELD_Y - 2 * SHIP_HEIGHT)
 | |
| #define ALIEN_START_Y (UFO_Y + 10)
 | |
| #define SCORENUM_Y SCORE_Y
 | |
| #define SCORENUM_X (PLAYFIELD_X + 6 * NUMBERS_WIDTH + 6 * NUM_SPACING)
 | |
| #define HISCORENUM_X (LCD_WIDTH - PLAYFIELD_X - 4 * NUMBERS_WIDTH - 3 * NUM_SPACING)
 | |
| #define SHIELD_Y (PLAYFIELD_Y - 5 * SHIP_HEIGHT)
 | |
| #define LIVES_X 8
 | |
| #define MAX_Y 15
 | |
| 
 | |
| 
 | |
| #else
 | |
|     #error INVADROX: Unsupported LCD type
 | |
| #endif
 | |
| 
 | |
| #define MAX_X ((LCD_WIDTH-LIVES_X*2-PLAYFIELD_X*2 - ALIEN_WIDTH)/2 - 1)
 | |
| 
 | |
| /* Defines common to each "graphic type" */
 | |
| #ifdef ARCADISH_GRAPHICS
 | |
| 
 | |
| #define SHOT_HEIGHT 5
 | |
| #define ALIEN_SPACING 4
 | |
| #define ALIEN_SPEED 2
 | |
| #define UFO_SPEED 1
 | |
| #define NUM_SPACING 3
 | |
| #define FIRE_SPEED 8
 | |
| #define BOMB_SPEED 3
 | |
| #define ALIENS 11
 | |
| 
 | |
| #elif defined SMALL_GRAPHICS
 | |
| 
 | |
| #define SHOT_HEIGHT 4
 | |
| #define ALIEN_SPACING 3
 | |
| #define ALIEN_SPEED 2
 | |
| #define UFO_SPEED 1
 | |
| #define NUM_SPACING 2
 | |
| #define FIRE_SPEED 6
 | |
| #define BOMB_SPEED 2
 | |
| #define ALIENS 11
 | |
| 
 | |
| #else
 | |
|     #error Graphic type not defined
 | |
| #endif
 | |
| 
 | |
| 
 | |
| /* Colors */
 | |
| #if (LCD_DEPTH >= 8)
 | |
| #define SLIME_GREEN LCD_RGBPACK(31, 254, 31)
 | |
| #define UFO_RED LCD_RGBPACK(254, 31, 31)
 | |
| #elif (LCD_DEPTH == 2)
 | |
| #define SLIME_GREEN LCD_LIGHTGRAY
 | |
| #define UFO_RED LCD_LIGHTGRAY
 | |
| #else
 | |
| #error LCD type not implemented yet
 | |
| #endif
 | |
| 
 | |
| /* Alien states */
 | |
| #define DEAD 0
 | |
| #define ALIVE 1
 | |
| #define BOMBER 2
 | |
| 
 | |
| /* Fire/bomb/ufo states */
 | |
| #define S_IDLE 0
 | |
| #define S_ACTIVE 1
 | |
| #define S_SHOWSCORE 2
 | |
| #define S_EXPLODE -9
 | |
| 
 | |
| /* Fire/bomb targets */
 | |
| #define TARGET_TOP 0
 | |
| #define TARGET_SHIELD 1
 | |
| #define TARGET_SHIP 2
 | |
| #define TARGET_BOTTOM 3
 | |
| #define TARGET_UFO 4
 | |
| 
 | |
| #define HISCOREFILE PLUGIN_GAMES_DATA_DIR "/invadrox.high"
 | |
| 
 | |
| 
 | |
| /* The time (in ms) for one iteration through the game loop - decrease this
 | |
|  * to speed up the game - note that current_tick is (currently) only accurate
 | |
|  * to 10ms.
 | |
|  */
 | |
| #define CYCLETIME 40
 | |
| 
 | |
| 
 | |
| /* Physical x is at PLAYFIELD_X + LIVES_X + x * ALIEN_SPEED
 | |
|  * Physical y is at y * ALIEN_HEIGHT
 | |
|  */
 | |
| struct alien {
 | |
|     int x;     /* x-coordinate (0 - 95) */
 | |
|     int y;     /* y-coordinate (0 - 18) */
 | |
|     unsigned char type;  /* 0 (Kang), 1 (Kodos), 2 (Serak) */
 | |
|     unsigned char state; /* Dead, alive or bomber */
 | |
| };
 | |
| 
 | |
| /* Aliens box 5 rows * ALIENS aliens in each row */
 | |
| struct alien aliens[5 * ALIENS];
 | |
| 
 | |
| #define MAX_BOMBS 4
 | |
| struct bomb {
 | |
|     int x, y;
 | |
|     unsigned char type;
 | |
|     unsigned char frame; /* Current animation frame */
 | |
|     unsigned char frames; /* Number of frames in animation */
 | |
|     unsigned char target; /* Remember target during explosion frames */
 | |
|     int state; /* 0 (IDLE) = inactive, 1 (FIRE) or negative, exploding */
 | |
| };
 | |
| struct bomb bombs[MAX_BOMBS];
 | |
| /* Increase max_bombs at higher levels */
 | |
| int max_bombs;
 | |
| 
 | |
| /* Raw framebuffer value of shield/ship green color */
 | |
| fb_data screen_green, screen_white;
 | |
| 
 | |
| /* For optimization, precalculate startoffset of each scanline */
 | |
| unsigned int ytab[LCD_HEIGHT];
 | |
| 
 | |
| int lives = 2;
 | |
| int score = 0;
 | |
| int scores[3] = { 30, 20, 10 };
 | |
| int level = 0;
 | |
| struct highscore hiscore;
 | |
| bool game_over = false;
 | |
| int ship_x, old_ship_x, ship_dir, ship_acc, max_ship_speed;
 | |
| int ship_frame, ship_frame_counter;
 | |
| bool ship_hit;
 | |
| int fire, fire_target, fire_x, fire_y;
 | |
| int curr_alien, aliens_paralyzed, gamespeed;
 | |
| int ufo_state, ufo_x;
 | |
| bool level_finished;
 | |
| bool aliens_down, aliens_right, hit_left_border, hit_right_border;
 | |
| 
 | |
| 
 | |
| /* No standard get_pixel function yet, use this hack instead */
 | |
| #if (LCD_DEPTH >= 8)
 | |
| 
 | |
| #if   defined(LCD_STRIDEFORMAT) && LCD_STRIDEFORMAT == VERTICAL_STRIDE
 | |
| static inline fb_data get_pixel(int x, int y)
 | |
| {
 | |
|     return rb->lcd_framebuffer[x*LCD_HEIGHT+y];
 | |
| }
 | |
| #else
 | |
| static inline fb_data get_pixel(int x, int y)
 | |
| {
 | |
|     return rb->lcd_framebuffer[ytab[y] + x];
 | |
| }
 | |
| #endif
 | |
| 
 | |
| #elif (LCD_DEPTH == 2)
 | |
| 
 | |
| #if (LCD_PIXELFORMAT == HORIZONTAL_PACKING)
 | |
| static const unsigned char shifts[4] = {
 | |
|     6, 4, 2, 0
 | |
| };
 | |
| /* Horizontal packing */
 | |
| static inline fb_data get_pixel(int x, int y)
 | |
| {
 | |
|     return (rb->lcd_framebuffer[ytab[y] + (x >> 2)] >> shifts[x & 3]) & 3;
 | |
| }
 | |
| #else
 | |
| /* Vertical packing */
 | |
| static const unsigned char shifts[4] = {
 | |
|     0, 2, 4, 6
 | |
| };
 | |
| static inline fb_data get_pixel(int x, int y)
 | |
| {
 | |
|     return (rb->lcd_framebuffer[ytab[y] + x] >> shifts[y & 3]) & 3;
 | |
| }
 | |
| #endif /* Horizontal/Vertical packing */
 | |
| 
 | |
| #else
 | |
|     #error get_pixel: pixelformat not implemented yet
 | |
| #endif
 | |
| 
 | |
| 
 | |
| /* Draw "digits" least significant digits of num at (x,y) */
 | |
| static void draw_number(int x, int y, int num, int digits)
 | |
| {
 | |
|     int i;
 | |
|     int d;
 | |
| 
 | |
|     for (i = digits - 1; i >= 0; i--) {
 | |
|         d = num % 10;
 | |
|         num = num / 10;
 | |
|         rb->lcd_bitmap_part(invadrox_numbers, d * NUMBERS_WIDTH, 0,
 | |
|                             STRIDE( SCREEN_MAIN, 
 | |
|                                     BMPWIDTH_invadrox_numbers, 
 | |
|                                     BMPHEIGHT_invadrox_numbers),
 | |
|                             x + i * (NUMBERS_WIDTH + NUM_SPACING), y,
 | |
|                             NUMBERS_WIDTH, FONT_HEIGHT);
 | |
|     }
 | |
|     /* Update lcd */
 | |
|     rb->lcd_update_rect(x, y, 4 * NUMBERS_WIDTH + 3 * NUM_SPACING, FONT_HEIGHT);
 | |
| }
 | |
| 
 | |
| 
 | |
| static inline void draw_score(void)
 | |
| {
 | |
|     draw_number(SCORENUM_X, SCORENUM_Y, score, 4);
 | |
|     if (score > hiscore.score) {
 | |
|         /* Draw new hiscore (same as score) */
 | |
|         draw_number(HISCORENUM_X, SCORENUM_Y, score, 4);
 | |
|     }
 | |
| }
 | |
| 
 | |
| 
 | |
| static void draw_level(void)
 | |
| {
 | |
|     draw_number(LEVEL_X + 2 * NUM_SPACING, PLAYFIELD_Y + 2, level, 2);
 | |
| }
 | |
| 
 | |
| 
 | |
| static void draw_lives(void)
 | |
| {
 | |
|     int i;
 | |
|     /* Lives num */
 | |
|     rb->lcd_bitmap_part(invadrox_numbers, lives * NUMBERS_WIDTH, 0,
 | |
|                         STRIDE( SCREEN_MAIN, 
 | |
|                                 BMPWIDTH_invadrox_numbers, 
 | |
|                                 BMPHEIGHT_invadrox_numbers), 
 | |
|                         PLAYFIELD_X + LIVES_X, PLAYFIELD_Y + 2,
 | |
|                         NUMBERS_WIDTH, FONT_HEIGHT);
 | |
| 
 | |
|     /* Ships */
 | |
|     for (i = 0; i < (lives - 1); i++) {
 | |
|         rb->lcd_bitmap_part(invadrox_ships, 0, 0, 
 | |
|                             STRIDE( SCREEN_MAIN, 
 | |
|                                     BMPWIDTH_invadrox_ships, 
 | |
|                                     BMPHEIGHT_invadrox_ships),
 | |
|                             PLAYFIELD_X + LIVES_X + SHIP_WIDTH + i * (SHIP_WIDTH + NUM_SPACING),
 | |
|                             PLAYFIELD_Y + 1, SHIP_WIDTH, SHIP_HEIGHT);
 | |
|     }
 | |
| 
 | |
|     /* Erase ship to the right (if less than MAX_LIVES) */
 | |
|     if (lives < MAX_LIVES) {
 | |
|         rb->lcd_fillrect(PLAYFIELD_X + LIVES_X + SHIP_WIDTH + i * (SHIP_WIDTH + NUM_SPACING),
 | |
|                          PLAYFIELD_Y + 1, SHIP_WIDTH, SHIP_HEIGHT);
 | |
|     }
 | |
|     /* Update lives (and level) part of screen */
 | |
|     rb->lcd_update_rect(PLAYFIELD_X + LIVES_X, PLAYFIELD_Y + 1,
 | |
|                         PLAYFIELD_WIDTH - 2 * LIVES_X, MAX(FONT_HEIGHT + 1, SHIP_HEIGHT + 1));
 | |
| }
 | |
| 
 | |
| 
 | |
| static inline void draw_aliens(void)
 | |
| {
 | |
|     int i;
 | |
| 
 | |
|     for (i = 0; i < 5 * ALIENS; i++) {
 | |
|         rb->lcd_bitmap_part(invadrox_aliens, aliens[i].x & 1 ? ALIEN_WIDTH : 0, 
 | |
|                             aliens[i].type * ALIEN_HEIGHT,
 | |
|                             STRIDE( SCREEN_MAIN, 
 | |
|                                     BMPWIDTH_invadrox_aliens, 
 | |
|                                     BMPHEIGHT_invadrox_aliens), 
 | |
|                             PLAYFIELD_X + LIVES_X + aliens[i].x * ALIEN_SPEED,
 | |
|                             ALIEN_START_Y + aliens[i].y * ALIEN_HEIGHT,
 | |
|                             ALIEN_WIDTH, ALIEN_HEIGHT);
 | |
|     }
 | |
| }
 | |
| 
 | |
| 
 | |
| /* Return false if there is no next alive alien (round is over) */
 | |
| static inline bool next_alien(void)
 | |
| {
 | |
|     bool ret = true;
 | |
| 
 | |
|     do {
 | |
|         curr_alien++;
 | |
|         if (curr_alien % ALIENS == 0) {
 | |
|             /* End of this row. Move up one row. */
 | |
|             curr_alien -= 2 * ALIENS;
 | |
|             if (curr_alien < 0) {
 | |
|                 /* No more aliens in this round. */
 | |
|                 curr_alien = 4 * ALIENS;
 | |
|                 ret = false;
 | |
|             }
 | |
|         }
 | |
|     } while (aliens[curr_alien].state == DEAD && ret);
 | |
| 
 | |
|     if (!ret) {
 | |
|         /* No more alive aliens. Round finished. */
 | |
|         if (hit_right_border) {
 | |
|             if (hit_left_border) {
 | |
|                 DBG("ERROR: both left and right borders are set (%d)\n", curr_alien);
 | |
|             }
 | |
|             /* Move down-left next round */
 | |
|             aliens_right = false;
 | |
|             aliens_down = true;
 | |
|             hit_right_border = false;
 | |
|         } else if (hit_left_border) {
 | |
|             /* Move down-right next round */
 | |
|             aliens_right = true;
 | |
|             aliens_down = true;
 | |
|             hit_left_border = false;
 | |
|         } else {
 | |
|             /* Not left nor right. Set down to false. */
 | |
|             aliens_down = false;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     return ret;
 | |
| }
 | |
| 
 | |
| 
 | |
| /* All aliens have been moved.
 | |
|  * Set curr_alien to first alive.
 | |
|  * Return false if no-one is left alive.
 | |
|  */
 | |
| static bool first_alien(void)
 | |
| {
 | |
|     int i, y;
 | |
| 
 | |
|     for (y = 4; y >= 0; y--) {
 | |
|         for (i = y * ALIENS; i < (y + 1) * ALIENS; i++) {
 | |
|             if (aliens[i].state != DEAD) {
 | |
|                 curr_alien = i;
 | |
|                 return true;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /* All aliens dead. */
 | |
|     level_finished = true;
 | |
| 
 | |
|     return false;
 | |
| }
 | |
| 
 | |
| 
 | |
| static bool move_aliens(void)
 | |
| {
 | |
|     int x, y, old_x, old_y;
 | |
| 
 | |
|     /* Move current alien (curr_alien is pointing to a living alien) */
 | |
| 
 | |
|     old_x = aliens[curr_alien].x;
 | |
|     old_y = aliens[curr_alien].y;
 | |
| 
 | |
|     if (aliens_down) {
 | |
|         aliens[curr_alien].y++;
 | |
|         if (aliens[curr_alien].y == MAX_Y) {
 | |
|             /* Alien is at bottom. Game Over. */
 | |
|             DBG("Alien %d is at bottom. Game Over.\n", curr_alien);
 | |
|             game_over = true;
 | |
|             return false;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (aliens_right) {
 | |
|         /* Moving right */
 | |
|         if (aliens[curr_alien].x < MAX_X) {
 | |
|             aliens[curr_alien].x++;
 | |
|         }
 | |
| 
 | |
|         /* Now, after move, check if we hit the right border. */
 | |
|         if (aliens[curr_alien].x == MAX_X) {
 | |
|             hit_right_border = true;
 | |
|         }
 | |
| 
 | |
|     } else {
 | |
|         /* Moving left */
 | |
|         if (aliens[curr_alien].x > 0) {
 | |
|             aliens[curr_alien].x--;
 | |
|         }
 | |
| 
 | |
|         /* Now, after move, check if we hit the left border. */
 | |
|         if (aliens[curr_alien].x == 0) {
 | |
|             hit_left_border = true;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /* Erase old position */
 | |
|     x = PLAYFIELD_X + LIVES_X + old_x * ALIEN_SPEED;
 | |
|     y = ALIEN_START_Y + old_y * ALIEN_HEIGHT;
 | |
|     if (aliens[curr_alien].y != old_y) {
 | |
|         /* Moved in y-dir. Erase whole alien. */
 | |
|         rb->lcd_fillrect(x, y, ALIEN_WIDTH, ALIEN_HEIGHT);
 | |
|     } else {
 | |
|         if (aliens_right) {
 | |
|             /* Erase left edge */
 | |
|             rb->lcd_fillrect(x, y, ALIEN_SPEED, ALIEN_HEIGHT);
 | |
|         } else {
 | |
|             /* Erase right edge */
 | |
|             x += ALIEN_WIDTH - ALIEN_SPEED;
 | |
|             rb->lcd_fillrect(x, y, ALIEN_SPEED, ALIEN_HEIGHT);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /* Draw alien at new pos */
 | |
|     x = PLAYFIELD_X + LIVES_X + aliens[curr_alien].x * ALIEN_SPEED;
 | |
|     y = ALIEN_START_Y + aliens[curr_alien].y * ALIEN_HEIGHT;
 | |
|     rb->lcd_bitmap_part(invadrox_aliens,
 | |
|                         aliens[curr_alien].x & 1 ? ALIEN_WIDTH : 0, 
 | |
|                         aliens[curr_alien].type * ALIEN_HEIGHT,
 | |
|                         STRIDE( SCREEN_MAIN, 
 | |
|                                 BMPWIDTH_invadrox_aliens,
 | |
|                                 BMPHEIGHT_invadrox_aliens), 
 | |
|                         x, y, ALIEN_WIDTH, ALIEN_HEIGHT);
 | |
| 
 | |
|     if (!next_alien()) {
 | |
|         /* Round finished. Set curr_alien to first alive from bottom. */
 | |
|         if (!first_alien()) {
 | |
|             /* Should never happen. Taken care of in move_fire(). */
 | |
|             return false;
 | |
|         }
 | |
|         /* TODO: Play next background sound */
 | |
|     }
 | |
| 
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| 
 | |
| static inline void draw_ship(void)
 | |
| {
 | |
|     /* Erase old ship */
 | |
|     if (old_ship_x < ship_x) {
 | |
|         /* Move right. Erase leftmost part of ship. */
 | |
|         rb->lcd_fillrect(old_ship_x, SHIP_Y, ship_x - old_ship_x, SHIP_HEIGHT);
 | |
|     } else if (old_ship_x > ship_x) {
 | |
|         /* Move left. Erase rightmost part of ship. */
 | |
|         rb->lcd_fillrect(ship_x + SHIP_WIDTH, SHIP_Y, old_ship_x - ship_x, SHIP_HEIGHT);
 | |
|     }
 | |
| 
 | |
|     /* Draw ship */
 | |
|     rb->lcd_bitmap_part(invadrox_ships, 0, ship_frame * SHIP_HEIGHT,
 | |
|                         STRIDE( SCREEN_MAIN, 
 | |
|                                 BMPWIDTH_invadrox_ships,
 | |
|                                 BMPHEIGHT_invadrox_ships), 
 | |
|                         ship_x, SHIP_Y, SHIP_WIDTH, SHIP_HEIGHT);
 | |
|     if (ship_hit) {
 | |
|         /* Alternate between frame 1 and 2 during hit */
 | |
|         ship_frame_counter++;
 | |
|         if (ship_frame_counter > 2) {
 | |
|             ship_frame_counter = 0;
 | |
|             ship_frame++;
 | |
|             if (ship_frame > 2) {
 | |
|                 ship_frame = 1;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /* Save ship_x for next time */
 | |
|     old_ship_x = ship_x;
 | |
| }
 | |
| 
 | |
| 
 | |
| static inline void fire_alpha(int xc, int yc, unsigned color)
 | |
| {
 | |
|     int oldmode = rb->lcd_get_drawmode();
 | |
| 
 | |
|     rb->lcd_set_foreground(color);
 | |
|     rb->lcd_set_drawmode(DRMODE_FG);
 | |
|     
 | |
|     rb->lcd_mono_bitmap(invadrox_fire, xc - (FIRE_WIDTH/2), yc, FIRE_WIDTH, FIRE_HEIGHT);
 | |
| 
 | |
|     rb->lcd_set_foreground(LCD_BLACK);
 | |
|     rb->lcd_set_drawmode(oldmode);
 | |
| }
 | |
| 
 | |
| 
 | |
| static void move_fire(void)
 | |
| {
 | |
|     bool hit_green = false;
 | |
|     bool hit_white = false;
 | |
|     int i, j;
 | |
|     static int exploding_alien = -1;
 | |
|     fb_data pix;
 | |
| 
 | |
|     if (fire == S_IDLE) {
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     /* Alien hit. Wait until explosion is finished. */
 | |
|     if (aliens_paralyzed < 0) {
 | |
|         aliens_paralyzed++;
 | |
|         if (aliens_paralyzed == 0) {
 | |
|             /* Erase exploding_alien */
 | |
|             rb->lcd_fillrect(PLAYFIELD_X + LIVES_X + aliens[exploding_alien].x * ALIEN_SPEED,
 | |
|                              ALIEN_START_Y + aliens[exploding_alien].y * ALIEN_HEIGHT,
 | |
|                              ALIEN_EXPLODE_WIDTH, ALIEN_HEIGHT);
 | |
|             fire = S_IDLE;
 | |
|             /* Special case. We killed curr_alien. */
 | |
|             if (exploding_alien == curr_alien) {
 | |
|                 if (!next_alien()) {
 | |
|                     /* Round finished. Set curr_alien to first alive from bottom. */
 | |
|                     first_alien();
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (fire == S_ACTIVE) {
 | |
| 
 | |
|         /* Erase */
 | |
|         rb->lcd_vline(fire_x, fire_y, fire_y + SHOT_HEIGHT);
 | |
| 
 | |
|         /* Check top */
 | |
|         if (fire_y <= SCORENUM_Y + FONT_HEIGHT + 4) {
 | |
| 
 | |
|             /* TODO: Play explode sound */
 | |
| 
 | |
|             fire = S_EXPLODE;
 | |
|             fire_target = TARGET_TOP;
 | |
|             fire_alpha(fire_x, fire_y, UFO_RED);
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         /* Move */
 | |
|         fire_y -= FIRE_SPEED;
 | |
| 
 | |
|         /* Hit UFO? */
 | |
|         if (ufo_state == S_ACTIVE) {
 | |
|             if ((ABS(ufo_x + UFO_WIDTH / 2 - fire_x) <= UFO_WIDTH / 2) &&
 | |
|                 (fire_y <= UFO_Y + UFO_HEIGHT)) {
 | |
|                 ufo_state = S_EXPLODE;
 | |
|                 fire = S_EXPLODE;
 | |
|                 fire_target = TARGET_UFO;
 | |
|                 /* Center explosion */
 | |
|                 ufo_x -= (UFO_EXPLODE_WIDTH - UFO_WIDTH) / 2;
 | |
|                 rb->lcd_bitmap(invadrox_ufo_explode, ufo_x, UFO_Y - 1,
 | |
|                                UFO_EXPLODE_WIDTH, UFO_EXPLODE_HEIGHT);
 | |
|                 return;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /* Hit bomb? (check position, not pixel value) */
 | |
|         for (i = 0; i < max_bombs; i++) {
 | |
|             if (bombs[i].state == S_ACTIVE) {
 | |
|                 /* Count as hit if within BOMB_WIDTH pixels */
 | |
|                 if ((ABS(bombs[i].x - fire_x) < BOMB_WIDTH) &&
 | |
|                     (fire_y - bombs[i].y < BOMB_HEIGHT)) {
 | |
|                     /* Erase bomb */
 | |
|                     rb->lcd_fillrect(bombs[i].x, bombs[i].y, BOMB_WIDTH, BOMB_HEIGHT);
 | |
|                     bombs[i].state = S_IDLE;
 | |
|                     /* Explode ship fire */
 | |
|                     fire = S_EXPLODE;
 | |
|                     fire_target = TARGET_SHIELD;
 | |
|                     fire_alpha(fire_x, fire_y, LCD_WHITE);
 | |
|                     return;
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /* Check for hit*/
 | |
|         for (i = FIRE_SPEED; i >= 0; i--) {
 | |
|             pix = get_pixel(fire_x, fire_y + i);
 | |
|             if(!memcmp(&pix, &screen_white, sizeof(fb_data))) {
 | |
|                 hit_white = true;
 | |
|                 fire_y += i;
 | |
|                 break;
 | |
|             }
 | |
|             if(!memcmp(&pix, &screen_green, sizeof(fb_data))) {
 | |
|                 hit_green = true;
 | |
|                 fire_y += i;
 | |
|                 break;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         if (hit_green) {
 | |
|             /* Hit shield */
 | |
| 
 | |
|             /* TODO: Play explode sound */
 | |
| 
 | |
|             fire = S_EXPLODE;
 | |
|             fire_target = TARGET_SHIELD;
 | |
|             /* Center explosion around hit pixel */
 | |
|             fire_y -= FIRE_HEIGHT / 2;
 | |
|             fire_alpha(fire_x, fire_y, SLIME_GREEN);
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         if (hit_white) {
 | |
| 
 | |
|             /* Hit alien? */
 | |
|             for (i = 0; i < 5 * ALIENS; i++) {
 | |
|                 if (aliens[i].state != DEAD &&
 | |
|                     (ABS(fire_x - (PLAYFIELD_X + LIVES_X + aliens[i].x * ALIEN_SPEED +
 | |
|                                    ALIEN_WIDTH / 2)) <= ALIEN_WIDTH / 2) &&
 | |
|                     (ABS(fire_y - (ALIEN_START_Y + aliens[i].y * ALIEN_HEIGHT +
 | |
|                                    ALIEN_HEIGHT / 2)) <= ALIEN_HEIGHT / 2)) {
 | |
| 
 | |
|                     /* TODO: play alien hit sound */
 | |
| 
 | |
|                     if (aliens[i].state == BOMBER) {
 | |
|                         /* Set (possible) alien above to bomber */
 | |
|                         for (j = i - ALIENS; j >= 0; j -= ALIENS) {
 | |
|                             if (aliens[j].state != DEAD) {
 | |
|                                 /* printf("New bomber (%d, %d)\n", j % ALIENS, j / ALIENS); */
 | |
|                                 aliens[j].state = BOMBER;
 | |
|                                 break;
 | |
|                             }
 | |
|                         }
 | |
|                     }
 | |
|                     aliens[i].state = DEAD;
 | |
|                     exploding_alien = i;
 | |
|                     score += scores[aliens[i].type];
 | |
|                     draw_score();
 | |
|                     /* Update score part of screen */
 | |
|                     rb->lcd_update_rect(SCORENUM_X, SCORENUM_Y,
 | |
|                                         PLAYFIELD_WIDTH - 2 * NUMBERS_WIDTH, FONT_HEIGHT);
 | |
| 
 | |
|                     /* Paralyze aliens S_EXPLODE frames */
 | |
|                     aliens_paralyzed = S_EXPLODE;
 | |
|                     rb->lcd_bitmap(invadrox_alien_explode,
 | |
|                                         PLAYFIELD_X + LIVES_X + aliens[i].x * ALIEN_SPEED,
 | |
|                                         ALIEN_START_Y + aliens[i].y * ALIEN_HEIGHT,
 | |
|                                         ALIEN_EXPLODE_WIDTH, ALIEN_EXPLODE_HEIGHT);
 | |
|                     /* Since alien is 1 pixel taller than explosion sprite, erase bottom line */
 | |
|                     rb->lcd_hline(PLAYFIELD_X + LIVES_X + aliens[i].x * ALIEN_SPEED,
 | |
|                                   PLAYFIELD_X + LIVES_X + aliens[i].x * ALIEN_SPEED + ALIEN_WIDTH,
 | |
|                                   ALIEN_START_Y + (aliens[i].y + 1) * ALIEN_HEIGHT - 1);
 | |
|                     return;
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /* Draw shot */
 | |
|         rb->lcd_set_foreground(LCD_WHITE);
 | |
|         rb->lcd_vline(fire_x, fire_y, fire_y + SHOT_HEIGHT);
 | |
|         rb->lcd_set_foreground(LCD_BLACK);
 | |
|     } else if (fire < S_IDLE) {
 | |
|         /* Count up towards S_IDLE, then erase explosion */
 | |
|         fire++;
 | |
|         if (fire == S_IDLE) {
 | |
|             /* Erase explosion */
 | |
|             if (fire_target == TARGET_TOP) {
 | |
|                 rb->lcd_fillrect(fire_x - (FIRE_WIDTH / 2), fire_y, FIRE_WIDTH, FIRE_HEIGHT);
 | |
|             } else if (fire_target == TARGET_SHIELD) {
 | |
|                 /* Draw explosion with black pixels */
 | |
|                 fire_alpha(fire_x, fire_y, LCD_BLACK);
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| 
 | |
| /* Return a BOMBER alien */
 | |
| static inline int random_bomber(void)
 | |
| {
 | |
|     int i, col;
 | |
| 
 | |
|     /* TODO: Weigh higher probability near ship */
 | |
|     col = rb->rand() % ALIENS;
 | |
|     for (i = col + 4 * ALIENS; i >= 0; i -= ALIENS) {
 | |
|         if (aliens[i].state == BOMBER) {
 | |
|             return i;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /* No BOMBER found in this col */
 | |
| 
 | |
|     for (i = 0; i < 5 * ALIENS; i++) {
 | |
|         if (aliens[i].state == BOMBER) {
 | |
|             return i;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /* No BOMBER found at all (error?) */
 | |
| 
 | |
|     return -1;
 | |
| }
 | |
| 
 | |
| 
 | |
| static inline void draw_bomb(int i)
 | |
| {
 | |
|     rb->lcd_bitmap_part(invadrox_bombs, bombs[i].type * BOMB_WIDTH,
 | |
|                         bombs[i].frame * BOMB_HEIGHT,
 | |
|                         STRIDE( SCREEN_MAIN, 
 | |
|                                 BMPWIDTH_invadrox_bombs,
 | |
|                                 BMPHEIGHT_invadrox_bombs), 
 | |
|                         bombs[i].x, bombs[i].y,
 | |
|                         BOMB_WIDTH, BOMB_HEIGHT);
 | |
|     /* Advance frame */
 | |
|     bombs[i].frame++;
 | |
|     if (bombs[i].frame == bombs[i].frames) {
 | |
|         bombs[i].frame = 0;
 | |
|     }
 | |
| }
 | |
| 
 | |
| 
 | |
| static void move_bombs(void)
 | |
| {
 | |
|     int i, j, bomber;
 | |
|     bool abort;
 | |
| 
 | |
|     for (i = 0; i < max_bombs; i++) {
 | |
| 
 | |
|         switch (bombs[i].state) {
 | |
| 
 | |
|         case S_IDLE:
 | |
|             if (ship_hit) {
 | |
|                 continue;
 | |
|             }
 | |
|             bomber = random_bomber();
 | |
|             if (bomber < 0) {
 | |
|                 DBG("ERROR: No bomber available\n");
 | |
|                 continue;
 | |
|             }
 | |
|             /* x, y */
 | |
|             bombs[i].x = PLAYFIELD_X + LIVES_X + aliens[bomber].x * ALIEN_SPEED + ALIEN_WIDTH / 2;
 | |
|             bombs[i].y = ALIEN_START_Y + (aliens[bomber].y + 1) * ALIEN_HEIGHT;
 | |
| 
 | |
|             /* Check for duplets in x and y direction */
 | |
|             abort = false;
 | |
|             for (j = i - 1; j >= 0; j--) {
 | |
|                 if ((bombs[j].state == S_ACTIVE) &&
 | |
|                     ((bombs[i].x == bombs[j].x) || (bombs[i].y == bombs[j].y))) {
 | |
|                     abort = true;
 | |
|                     break;
 | |
|                 }
 | |
|             }
 | |
|             if (abort) {
 | |
|                 /* Skip this one, continue with next bomb */
 | |
|                 /* printf("Bomb %d duplet of %d\n", i, j); */
 | |
|                 continue;
 | |
|             }
 | |
| 
 | |
|             /* Passed, set type */
 | |
|             bombs[i].type = rb->rand() % 3;
 | |
|             bombs[i].frame = 0;
 | |
|             if (bombs[i].type == 0) {
 | |
|                 bombs[i].frames = 3;
 | |
|             } else if (bombs[i].type == 1) {
 | |
|                 bombs[i].frames = 4;
 | |
|             } else {
 | |
|                 bombs[i].frames = 6;
 | |
|             }
 | |
| 
 | |
|             /* Bombs away */
 | |
|             bombs[i].state = S_ACTIVE;
 | |
|             draw_bomb(i);
 | |
|             continue;
 | |
| 
 | |
|             break;
 | |
| 
 | |
|         case S_ACTIVE:
 | |
|             /* Erase old position */
 | |
|             rb->lcd_fillrect(bombs[i].x, bombs[i].y, BOMB_WIDTH, BOMB_HEIGHT);
 | |
| 
 | |
|             /* Move */
 | |
|             bombs[i].y += BOMB_SPEED;
 | |
| 
 | |
|             /* Check if bottom hit */
 | |
|             if (bombs[i].y + BOMB_HEIGHT >= PLAYFIELD_Y) {
 | |
|                 bombs[i].y = PLAYFIELD_Y - FIRE_HEIGHT + 1;
 | |
|                 fire_alpha(bombs[i].x, bombs[i].y, LCD_WHITE);
 | |
|                 bombs[i].state = S_EXPLODE;
 | |
|                 bombs[i].target = TARGET_BOTTOM;
 | |
|                 break;
 | |
|             }
 | |
| 
 | |
|             /* Check for green (ship or shield) */
 | |
|             for (j = BOMB_HEIGHT; j >= BOMB_HEIGHT - BOMB_SPEED; j--) {
 | |
|                 bombs[i].target = 0;
 | |
|                 fb_data pix = get_pixel(bombs[i].x + BOMB_WIDTH / 2, bombs[i].y + j);
 | |
|                 if(!memcmp(&pix, &screen_green, sizeof(fb_data))) {
 | |
|                     /* Move to hit pixel */
 | |
|                     bombs[i].x += BOMB_WIDTH / 2;
 | |
|                     bombs[i].y += j;
 | |
| 
 | |
|                     /* Check if ship is hit */
 | |
|                     if (bombs[i].y > SHIELD_Y + SHIELD_HEIGHT && bombs[i].y < PLAYFIELD_Y) {
 | |
| 
 | |
|                         /* TODO: play ship hit sound */
 | |
| 
 | |
|                         ship_hit = true;
 | |
|                         ship_frame = 1;
 | |
|                         ship_frame_counter = 0;
 | |
|                         bombs[i].state = S_EXPLODE * 4;
 | |
|                         bombs[i].target = TARGET_SHIP;
 | |
|                         rb->lcd_bitmap_part(invadrox_ships, 0, 1 * SHIP_HEIGHT,
 | |
|                                             STRIDE( SCREEN_MAIN, 
 | |
|                                                     BMPWIDTH_invadrox_ships,
 | |
|                                                     BMPHEIGHT_invadrox_ships), 
 | |
|                                             ship_x, SHIP_Y,
 | |
|                                             SHIP_WIDTH, SHIP_HEIGHT);
 | |
|                         break;
 | |
|                     }
 | |
|                     /* Shield hit */
 | |
|                     bombs[i].state = S_EXPLODE;
 | |
|                     bombs[i].target = TARGET_SHIELD;
 | |
|                     /* Center explosion around hit pixel in shield */
 | |
|                     bombs[i].y -= FIRE_HEIGHT / 2;
 | |
|                     fire_alpha(bombs[i].x, bombs[i].y, SLIME_GREEN);
 | |
|                     break;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             if (bombs[i].target != 0) {
 | |
|                 /* Hit ship or shield, continue */
 | |
|                 continue;
 | |
|             }
 | |
| 
 | |
|             draw_bomb(i);
 | |
|             break;
 | |
| 
 | |
|         default:
 | |
|             /* If we get here state should be < 0, exploding */
 | |
|             bombs[i].state++;
 | |
|             if (bombs[i].state == S_IDLE) {
 | |
|                 if (ship_hit) {
 | |
|                     /* Erase explosion */
 | |
|                     rb->lcd_fillrect(ship_x, SHIP_Y, SHIP_WIDTH, SHIP_HEIGHT);
 | |
|                     rb->lcd_update_rect(ship_x, SHIP_Y, SHIP_WIDTH, SHIP_HEIGHT);
 | |
|                     ship_hit = false;
 | |
|                     ship_frame = 0;
 | |
|                     ship_x = PLAYFIELD_X + 2 * LIVES_X;
 | |
|                     lives--;
 | |
|                     if (lives == 0) {
 | |
|                         game_over = true;
 | |
|                         return;
 | |
|                     }
 | |
|                     draw_lives();
 | |
|                     /* Sleep 1s to give player time to examine lives left */
 | |
|                     rb->sleep(HZ);
 | |
|                 }
 | |
|                 /* Erase explosion (even if ship hit, might be another bomb) */
 | |
|                 fire_alpha(bombs[i].x, bombs[i].y, LCD_BLACK);
 | |
|             }
 | |
|             break;
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| 
 | |
| static inline void move_ship(void)
 | |
| {
 | |
|     ship_dir += ship_acc;
 | |
|     if (ship_dir > max_ship_speed) {
 | |
|         ship_dir = max_ship_speed;
 | |
|     }
 | |
|     if (ship_dir < -max_ship_speed) {
 | |
|         ship_dir = -max_ship_speed;
 | |
|     }
 | |
|     ship_x += ship_dir;
 | |
|     if (ship_x < SHIP_MIN_X) {
 | |
|         ship_x = SHIP_MIN_X;
 | |
|     }
 | |
|     if (ship_x > SHIP_MAX_X) {
 | |
|         ship_x = SHIP_MAX_X;
 | |
|     }
 | |
| 
 | |
|     draw_ship();
 | |
| }
 | |
| 
 | |
| 
 | |
| /* Unidentified Flying Object */
 | |
| static void move_ufo(void)
 | |
| {
 | |
|     static int ufo_speed;
 | |
|     static int counter;
 | |
|     int mystery_score;
 | |
| 
 | |
|     switch (ufo_state) {
 | |
| 
 | |
|     case S_IDLE:
 | |
| 
 | |
|         if (rb->rand() % 500 == 0) {
 | |
|             /* Uh-oh, it's time to launch a mystery UFO */
 | |
| 
 | |
|             /* TODO: Play UFO sound */
 | |
| 
 | |
|             if (rb->rand() % 2) {
 | |
|                 ufo_speed = UFO_SPEED;
 | |
|                 ufo_x = PLAYFIELD_X;
 | |
|             } else {
 | |
|                 ufo_speed = -UFO_SPEED;
 | |
|                 ufo_x = LCD_WIDTH - PLAYFIELD_X - UFO_WIDTH;
 | |
|             }
 | |
|             ufo_state = S_ACTIVE;
 | |
|             /* UFO will be drawn next frame */
 | |
|         }
 | |
|         break;
 | |
| 
 | |
|     case S_ACTIVE:
 | |
|         /* Erase old pos */
 | |
|         rb->lcd_fillrect(ufo_x, UFO_Y, UFO_WIDTH, UFO_HEIGHT);
 | |
|         /* Move */
 | |
|         ufo_x += ufo_speed;
 | |
|         /* Check bounds */
 | |
|         if (ufo_x < PLAYFIELD_X || ufo_x > LCD_WIDTH - PLAYFIELD_X - UFO_WIDTH) {
 | |
|             ufo_state = S_IDLE;
 | |
|             break;
 | |
|         }
 | |
|         /* Draw new pos */
 | |
|         rb->lcd_bitmap(invadrox_ufo, ufo_x, UFO_Y, UFO_WIDTH, UFO_HEIGHT);
 | |
|         break;
 | |
| 
 | |
|     case S_SHOWSCORE:
 | |
|         counter++;
 | |
|         if (counter == S_IDLE) {
 | |
|             /* Erase mystery number */
 | |
|             rb->lcd_fillrect(ufo_x, UFO_Y, 3 * NUMBERS_WIDTH + 2 * NUM_SPACING, FONT_HEIGHT);
 | |
|             ufo_state = S_IDLE;
 | |
|         }
 | |
|         break;
 | |
| 
 | |
|     default:
 | |
|         /* Exploding */
 | |
|         ufo_state++;
 | |
|         if (ufo_state == S_IDLE) {
 | |
|             /* Erase explosion */
 | |
|             rb->lcd_fillrect(ufo_x, UFO_Y - 1, UFO_EXPLODE_WIDTH, UFO_EXPLODE_HEIGHT);
 | |
|             ufo_state = S_SHOWSCORE;
 | |
|             counter = S_EXPLODE * 4;
 | |
|             /* Draw mystery_score, sleep, increase score and continue */
 | |
|             mystery_score = 50 + (rb->rand() % 6) * 50;
 | |
|             if (mystery_score < 100) {
 | |
|                 draw_number(ufo_x, UFO_Y, mystery_score, 2);
 | |
|             } else {
 | |
|                 draw_number(ufo_x, UFO_Y, mystery_score, 3);
 | |
|             }
 | |
|             score += mystery_score;
 | |
|             draw_score();
 | |
|         }
 | |
|         break;
 | |
|     }
 | |
| }
 | |
| 
 | |
| 
 | |
| static void draw_background(void)
 | |
| {
 | |
| 
 | |
|     rb->lcd_bitmap(invadrox_background, 0, 0, LCD_WIDTH, LCD_HEIGHT);
 | |
|     rb->lcd_update();
 | |
| }
 | |
| 
 | |
| 
 | |
| static void new_level(void)
 | |
| {
 | |
|     int i;
 | |
| 
 | |
|     draw_background();
 | |
|     /* Give an extra life for each new level */
 | |
|     if (lives < MAX_LIVES) {
 | |
|         lives++;
 | |
|     }
 | |
|     draw_lives();
 | |
| 
 | |
|     /* Score */
 | |
|     draw_score();
 | |
|     draw_number(HISCORENUM_X, SCORENUM_Y, hiscore.score, 4);
 | |
| 
 | |
|     level++;
 | |
|     draw_level();
 | |
|     level_finished = false;
 | |
| 
 | |
|     ufo_state = S_IDLE;
 | |
| 
 | |
|     /* Init alien positions and states */
 | |
|     for (i = 0; i < 4 * ALIENS; i++) {
 | |
|         aliens[i].x = 0 + (i % ALIENS) * ((ALIEN_WIDTH + ALIEN_SPACING) / ALIEN_SPEED);
 | |
|         aliens[i].y = 2 * (i / ALIENS);
 | |
|         aliens[i].state = ALIVE;
 | |
|     }
 | |
|     /* Last row, bombers */
 | |
|     for (i = 4 * ALIENS; i < 5 * ALIENS; i++) {
 | |
|         aliens[i].x = 0 + (i % ALIENS) * ((ALIEN_WIDTH + ALIEN_SPACING) / ALIEN_SPEED);
 | |
|         aliens[i].y = 2 * (i / ALIENS);
 | |
|         aliens[i].state = BOMBER;
 | |
|     }
 | |
| 
 | |
|     /* Init bombs to inactive (S_IDLE) */
 | |
|     for (i = 0; i < MAX_BOMBS; i++) {
 | |
|         bombs[i].state = S_IDLE;
 | |
|     }
 | |
| 
 | |
|     /* Start aliens closer to earth from level 2 */
 | |
|     for (i = 0; i < 5 * ALIENS; i++) {
 | |
|         if (level < 6) {
 | |
|             aliens[i].y += level - 1;
 | |
|         } else {
 | |
|             aliens[i].y += 5;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /* Max concurrent bombs */
 | |
|     max_bombs = 1;
 | |
| 
 | |
|     gamespeed = 2;
 | |
| 
 | |
|     if (level > 1) {
 | |
|         max_bombs++;
 | |
|     }
 | |
| 
 | |
|     /* Increase speed */
 | |
|     if (level > 2) {
 | |
|         gamespeed++;
 | |
|     }
 | |
| 
 | |
|     if (level > 3) {
 | |
|         max_bombs++;
 | |
|     }
 | |
| 
 | |
|     /* Increase speed more */
 | |
|     if (level > 4) {
 | |
|         gamespeed++;
 | |
|     }
 | |
| 
 | |
|     if (level > 5) {
 | |
|         max_bombs++;
 | |
|     }
 | |
| 
 | |
|     /* 4 shields */
 | |
|     for (i = 1; i <= 4; i++) {
 | |
|         rb->lcd_bitmap(invadrox_shield,
 | |
|                             PLAYFIELD_X + i * PLAYFIELD_WIDTH / 5 - SHIELD_WIDTH / 2,
 | |
|                             SHIELD_Y, SHIELD_WIDTH, SHIELD_HEIGHT);
 | |
|     }
 | |
| 
 | |
|     /* Bottom line */
 | |
|     rb->lcd_set_foreground(SLIME_GREEN);
 | |
|     rb->lcd_hline(PLAYFIELD_X, LCD_WIDTH - PLAYFIELD_X, PLAYFIELD_Y);
 | |
|     /* Restore foreground to black (for fast erase later). */
 | |
|     rb->lcd_set_foreground(LCD_BLACK);
 | |
| 
 | |
|     ship_x = PLAYFIELD_X + 2 * LIVES_X;
 | |
|     if (level == 1) {
 | |
|         old_ship_x = ship_x;
 | |
|     }
 | |
|     ship_dir = 0;
 | |
|     ship_acc = 0;
 | |
|     ship_frame = 0;
 | |
|     ship_hit = false;
 | |
|     fire = S_IDLE;
 | |
|     /* Start moving the bottom row left to right */
 | |
|     curr_alien = 4 * ALIENS;
 | |
|     aliens_paralyzed = 0;
 | |
|     aliens_right = true;
 | |
|     aliens_down = false;
 | |
|     hit_left_border = false;
 | |
|     hit_right_border = false;
 | |
|     /* TODO: Change max_ship_speed to 3 at higher levels */
 | |
|     max_ship_speed = 2;
 | |
| 
 | |
|     draw_aliens();
 | |
| 
 | |
|     rb->lcd_update();
 | |
| }
 | |
| 
 | |
| 
 | |
| static void init_invadrox(void)
 | |
| {
 | |
|     int i;
 | |
| 
 | |
|     /* Seed random number generator with a "random" number */
 | |
|     rb->srand(rb->get_time()->tm_sec + rb->get_time()->tm_min * 60);
 | |
| 
 | |
|     /* Precalculate start of each scanline */
 | |
|     for (i = 0; i < LCD_HEIGHT; i++) {
 | |
| #if (LCD_DEPTH >= 8)
 | |
|         ytab[i] = i * LCD_WIDTH;
 | |
| #elif (LCD_DEPTH == 2) && (LCD_PIXELFORMAT == HORIZONTAL_PACKING)
 | |
|         ytab[i] = i * (LCD_WIDTH / 4);
 | |
| #elif (LCD_DEPTH == 2) && (LCD_PIXELFORMAT == VERTICAL_PACKING)
 | |
|         ytab[i] = (i / 4) * LCD_WIDTH;
 | |
| #else
 | |
|         #error pixelformat not implemented yet
 | |
| #endif
 | |
|     }
 | |
| 
 | |
|     rb->lcd_set_background(LCD_BLACK);
 | |
|     rb->lcd_set_foreground(LCD_BLACK);
 | |
| 
 | |
|     if (highscore_load(HISCOREFILE, &hiscore, 1) < 0) {
 | |
|         /* Init hiscore to 0 */
 | |
|         rb->strlcpy(hiscore.name, "Invader", sizeof(hiscore.name));
 | |
|         hiscore.score = 0;
 | |
|         hiscore.level = 1;
 | |
|     }
 | |
| 
 | |
|     /* Init alien types in aliens array */
 | |
|     for (i = 0; i < 1 * ALIENS; i++) {
 | |
|         aliens[i].type = 0; /* Kang */
 | |
|     }
 | |
|     for (; i < 3 * ALIENS; i++) {
 | |
|         aliens[i].type = 1; /* Kodos */
 | |
|     }
 | |
|     for (; i < 5 * ALIENS; i++) {
 | |
|         aliens[i].type = 2; /* Serak */
 | |
|     }
 | |
| 
 | |
| 
 | |
|     /* Save screen white color */
 | |
|     rb->lcd_set_foreground(LCD_WHITE);
 | |
|     rb->lcd_drawpixel(0, 0);
 | |
|     rb->lcd_update_rect(0, 0, 1, 1);
 | |
|     screen_white = get_pixel(0, 0);
 | |
| 
 | |
|     /* Save screen green color */
 | |
|     rb->lcd_set_foreground(SLIME_GREEN);
 | |
|     rb->lcd_drawpixel(0, 0);
 | |
|     rb->lcd_update_rect(0, 0, 1, 1);
 | |
|     screen_green = get_pixel(0, 0);
 | |
| 
 | |
|     /* Restore black foreground */
 | |
|     rb->lcd_set_foreground(LCD_BLACK);
 | |
| 
 | |
|     new_level();
 | |
| 
 | |
|     /* Flash score at start */
 | |
|     for (i = 0; i < 5; i++) {
 | |
|         rb->lcd_fillrect(SCORENUM_X, SCORENUM_Y,
 | |
|                          4 * NUMBERS_WIDTH + 3 * NUM_SPACING,
 | |
|                          FONT_HEIGHT);
 | |
|         rb->lcd_update_rect(SCORENUM_X, SCORENUM_Y,
 | |
|                             4 * NUMBERS_WIDTH + 3 * NUM_SPACING,
 | |
|                             FONT_HEIGHT);
 | |
|         rb->sleep(HZ / 10);
 | |
|         draw_number(SCORENUM_X, SCORENUM_Y, score, 4);
 | |
|         rb->sleep(HZ / 10);
 | |
|     }
 | |
| }
 | |
| 
 | |
| 
 | |
| static inline bool handle_buttons(void)
 | |
| {
 | |
|     static unsigned int oldbuttonstate = 0;
 | |
| 
 | |
|     unsigned int released, pressed, newbuttonstate;
 | |
| 
 | |
|     if (ship_hit) {
 | |
|         /* Don't allow ship movement during explosion */
 | |
|         newbuttonstate = 0;
 | |
|     } else {
 | |
|         newbuttonstate = rb->button_status();
 | |
|     }
 | |
|     if(newbuttonstate == oldbuttonstate) {
 | |
|         if (newbuttonstate == 0) {
 | |
|             /* No button pressed. Stop ship. */
 | |
|             ship_acc = 0;
 | |
|             if (ship_dir > 0) {
 | |
|                 ship_dir--;
 | |
|             }
 | |
|             if (ship_dir < 0) {
 | |
|                 ship_dir++;
 | |
|             }
 | |
|         }
 | |
|         /* return false; */
 | |
|         goto check_usb;
 | |
|     }
 | |
|     released = ~newbuttonstate & oldbuttonstate;
 | |
|     pressed = newbuttonstate & ~oldbuttonstate;
 | |
|     oldbuttonstate = newbuttonstate;
 | |
|     if (pressed) {
 | |
|         if (pressed & ACTION_LEFT) {
 | |
|             if (ship_acc > -1) {
 | |
|                 ship_acc--;
 | |
|             }
 | |
|         }
 | |
|         if (pressed & ACTION_RIGHT) {
 | |
|             if (ship_acc < 1) {
 | |
|                 ship_acc++;
 | |
|             }
 | |
|         }
 | |
|         if (pressed & ACTION_FIRE) {
 | |
|             if (fire == S_IDLE) {
 | |
|                 /* Fire shot */
 | |
|                 fire_x = ship_x + SHIP_WIDTH / 2;
 | |
|                 fire_y = SHIP_Y - SHOT_HEIGHT;
 | |
|                 fire = S_ACTIVE;
 | |
|                 /* TODO: play fire sound */
 | |
|             }
 | |
|         }
 | |
|         if (pressed & ACTION_QUIT) {
 | |
|             rb->splash(HZ * 1, "Quit");
 | |
|             return true;
 | |
|         }
 | |
|     }
 | |
|     if (released) {
 | |
|         if ((released & ACTION_LEFT)) {
 | |
|             if (ship_acc < 1) {
 | |
|                 ship_acc++;
 | |
|             }
 | |
|         }
 | |
|         if ((released & ACTION_RIGHT)) {
 | |
|             if (ship_acc > -1) {
 | |
|                 ship_acc--;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
| check_usb:
 | |
| 
 | |
|     /* Quit if USB is connected */
 | |
|     if (rb->button_get(false) == SYS_USB_CONNECTED) {
 | |
|         return true;
 | |
|     }
 | |
| 
 | |
|     return false;
 | |
| }
 | |
| 
 | |
| 
 | |
| static void game_loop(void)
 | |
| {
 | |
|     int i, end;
 | |
| 
 | |
|     /* Print dimensions (just for debugging) */
 | |
|     DBG("%03dx%03dx%02d\n", LCD_WIDTH, LCD_HEIGHT, LCD_DEPTH);
 | |
| 
 | |
|     /* Init */
 | |
|     init_invadrox();
 | |
| 
 | |
|     while (1) {
 | |
|         /* Convert CYCLETIME (in ms) to HZ */
 | |
|         end = *rb->current_tick + (CYCLETIME * HZ) / 1000;
 | |
| 
 | |
|         if (handle_buttons()) {
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         /* Animate */
 | |
|         move_ship();
 | |
|         move_fire();
 | |
| 
 | |
|         /* Check if level is finished (marked by move_fire) */
 | |
|         if (level_finished) {
 | |
|             /* TODO: Play level finished sound */
 | |
|             new_level();
 | |
|         }
 | |
| 
 | |
|         move_ufo();
 | |
| 
 | |
|         /* Move aliens */
 | |
|         if (!aliens_paralyzed && !ship_hit) {
 | |
|             for (i = 0; i < gamespeed; i++) {
 | |
|                 if (!move_aliens()) {
 | |
|                     if (game_over) {
 | |
|                         return;
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /* Move alien bombs */
 | |
|         move_bombs();
 | |
|         if (game_over) {
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         /* Update "playfield" rect */
 | |
|         rb->lcd_update_rect(PLAYFIELD_X, SCORENUM_Y + FONT_HEIGHT,
 | |
|                             PLAYFIELD_WIDTH,
 | |
|                             PLAYFIELD_Y + 1 - SCORENUM_Y - FONT_HEIGHT);
 | |
| 
 | |
|         /* Wait until next frame */
 | |
|         DBG("%ld (%d)\n", end - *rb->current_tick, (CYCLETIME * HZ) / 1000);
 | |
|         if (TIME_BEFORE(*rb->current_tick, end)) {
 | |
|             rb->sleep(end - *rb->current_tick);
 | |
|         } else {
 | |
|             rb->yield();
 | |
|         }
 | |
| 
 | |
|     } /* end while */
 | |
| }
 | |
| 
 | |
| 
 | |
| /* this is the plugin entry point */
 | |
| enum plugin_status plugin_start(UNUSED const void* parameter)
 | |
| {
 | |
|     rb->lcd_setfont(FONT_SYSFIXED);
 | |
|     /* Turn off backlight timeout */
 | |
|     backlight_ignore_timeout();
 | |
| 
 | |
|     /* now go ahead and have fun! */
 | |
|     game_loop();
 | |
| 
 | |
|     /* Game Over. */
 | |
|     /* TODO: Play game over sound */
 | |
|     rb->splash(HZ * 2, "Game Over");
 | |
|     if (score > hiscore.score) {
 | |
|         /* Save new hiscore */
 | |
|         highscore_update(score, level, "Invader", &hiscore, 1);
 | |
|         highscore_save(HISCOREFILE, &hiscore, 1);
 | |
|     }
 | |
| 
 | |
|     /* Restore user's original backlight setting */
 | |
|     rb->lcd_setfont(FONT_UI);
 | |
|     /* Turn on backlight timeout (revert to settings) */
 | |
|     backlight_use_settings();
 | |
| 
 | |
|     return PLUGIN_OK;
 | |
| }
 | |
| 
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * GNU Emacs settings: Kernighan & Richie coding style with
 | |
|  *                     4 spaces indent and no tabs.
 | |
|  * Local Variables:
 | |
|  *  c-file-style: "k&r"
 | |
|  *  c-basic-offset: 4
 | |
|  *  indent-tabs-mode: nil
 | |
|  * End:
 | |
|  */
 |