2011-06-30 15:17:21 +02:00
|
|
|
#include <stdio.h>
|
2011-06-30 16:15:03 +02:00
|
|
|
#include <string.h>
|
|
|
|
#include <stdlib.h>
|
2011-06-30 15:17:21 +02:00
|
|
|
|
2011-07-01 12:19:23 +02:00
|
|
|
/* levels have fixed, hardcoded dimensions */
|
2011-06-30 15:17:21 +02:00
|
|
|
#define LEVEL_HEIGHT 9
|
|
|
|
#define LEVEL_WIDTH 16
|
2011-06-30 16:15:03 +02:00
|
|
|
|
2011-07-01 12:27:57 +02:00
|
|
|
typedef enum { WALL='#', BALL='@', CUBE='H', VOID=' ', GIFT='*' } cell;
|
|
|
|
typedef enum { UP='u', DOWN='d', LEFT='l', RIGHT='r' } direction;
|
2011-07-01 12:19:23 +02:00
|
|
|
|
|
|
|
char* levels[] = { "################"
|
|
|
|
"#@## *#H#"
|
|
|
|
"# * ###"
|
|
|
|
"# ##* #"
|
|
|
|
"# ## * ##"
|
|
|
|
"## * * * #"
|
|
|
|
"# * *## * #"
|
|
|
|
"# ##* *#"
|
|
|
|
"################",
|
|
|
|
|
|
|
|
" # # # # # ##"
|
|
|
|
"# * @#"
|
|
|
|
" #* #* * "
|
|
|
|
"# # * * # #"
|
|
|
|
" # * # "
|
|
|
|
"# #H# * #"
|
|
|
|
" # # # #**#"
|
|
|
|
"# # "
|
|
|
|
" # # # ",
|
|
|
|
|
|
|
|
"################"
|
|
|
|
"# *#@#"
|
|
|
|
"# ## ##H#"
|
|
|
|
"# #* * #"
|
|
|
|
"# * *## *#"
|
|
|
|
"# #* * *# *##"
|
|
|
|
"# ##* #* * *###"
|
|
|
|
"#* ##* #"
|
|
|
|
"################",
|
|
|
|
|
|
|
|
"################"
|
|
|
|
"# #H#"
|
|
|
|
"# # #"
|
|
|
|
"##*#* *#*#*#*#*#"
|
|
|
|
"# # #* *# # # ##"
|
|
|
|
"##*#*#* *#*#*#*#"
|
|
|
|
"# # #"
|
|
|
|
"# # #@ #"
|
|
|
|
"################",
|
|
|
|
|
|
|
|
" ############## "
|
|
|
|
"#@ # # # #"
|
|
|
|
"# #* # * * # #"
|
|
|
|
"## # # #"
|
|
|
|
"#* #*# ##"
|
|
|
|
"## # * # #"
|
|
|
|
"#*# # # # #H#"
|
|
|
|
"# # *# #*#"
|
|
|
|
" ############## ",
|
|
|
|
|
|
|
|
" ############"
|
|
|
|
" # * #* *#"
|
|
|
|
" # * # ##"
|
|
|
|
" # * #"
|
|
|
|
"#@ * #"
|
|
|
|
"## * # ##"
|
|
|
|
"# * # #"
|
|
|
|
"#H # * ##* #"
|
|
|
|
"################",
|
|
|
|
|
|
|
|
"################"
|
|
|
|
"# #"
|
|
|
|
" ## ### #* ##*#"
|
|
|
|
" #* #* # # # # "
|
|
|
|
" # # ### ## "
|
|
|
|
" ## # #*# #*# "
|
|
|
|
"# #"
|
|
|
|
"# @#* H #*#"
|
|
|
|
"################",
|
|
|
|
|
|
|
|
"############### "
|
|
|
|
"# *## ##"
|
|
|
|
"# #* ## * #"
|
|
|
|
"# *## # #* #"
|
|
|
|
"## ## #*# #"
|
|
|
|
"## # *#* #"
|
|
|
|
"#*H*# * #@# #"
|
|
|
|
"## #"
|
|
|
|
" ###############",
|
|
|
|
|
|
|
|
" # ########### "
|
|
|
|
" #*#* # @#"
|
|
|
|
"#* *# * # "
|
|
|
|
" # # *## *# #"
|
|
|
|
"# #* #*H* *#"
|
|
|
|
"# *## # "
|
|
|
|
"#*#* # "
|
|
|
|
"# # "
|
|
|
|
"############ ",
|
|
|
|
|
|
|
|
" ########### "
|
|
|
|
"#### * #"
|
|
|
|
"# H ###* *# *#"
|
|
|
|
"# * #* #* # "
|
|
|
|
"# # * # *#"
|
|
|
|
"#*#* # *# #@# "
|
|
|
|
" #* ### ### "
|
|
|
|
"# # # # "
|
|
|
|
" ######### # #",
|
|
|
|
|
|
|
|
"################"
|
|
|
|
"# # @#"
|
|
|
|
"# #** ** ##"
|
|
|
|
"## * ## *#"
|
|
|
|
"#* #*#** ###"
|
|
|
|
"## ## ## #"
|
|
|
|
"#* *# * H *#"
|
|
|
|
"##*### # ##"
|
|
|
|
" ## ########### ",
|
|
|
|
|
|
|
|
"## ## #### "
|
|
|
|
"#@#####* ### *##"
|
|
|
|
"# ** * #"
|
|
|
|
"# ## ##* #*# #"
|
|
|
|
"# # * ###* ## #"
|
|
|
|
"# ## ## #H# #"
|
|
|
|
"# * #"
|
|
|
|
"# * #"
|
|
|
|
"################",
|
|
|
|
|
|
|
|
" ############## "
|
|
|
|
"# @# * ##"
|
|
|
|
"# # #* *## #"
|
|
|
|
"# * # #"
|
|
|
|
"# * #*#"
|
|
|
|
"# # * #"
|
|
|
|
"## * * #*#"
|
|
|
|
"#H # * # # #"
|
|
|
|
" ############## ",
|
|
|
|
|
|
|
|
"################"
|
|
|
|
"#*#* *#*#"
|
|
|
|
"# *#@ ## #"
|
|
|
|
"# H * #"
|
|
|
|
"# *# #"
|
|
|
|
"# * #"
|
|
|
|
"# *# # #"
|
|
|
|
"#*#* *#*#"
|
|
|
|
"################",
|
|
|
|
|
|
|
|
" ###### ####### "
|
|
|
|
"# *# * #"
|
|
|
|
"# # * # # * #"
|
|
|
|
"# @# #** #* #"
|
|
|
|
" # # # * H# #"
|
|
|
|
"#* # #* #"
|
|
|
|
" # * # "
|
|
|
|
"#* *#"
|
|
|
|
" ############## ",
|
|
|
|
|
|
|
|
"################"
|
|
|
|
"## H#* * *#"
|
|
|
|
"#* @*#* ##"
|
|
|
|
"## ### * ##"
|
|
|
|
"## *#*# #"
|
|
|
|
"#** *#* #"
|
|
|
|
"## * ####* #"
|
|
|
|
"##*# # #"
|
|
|
|
"################",
|
|
|
|
|
|
|
|
"################"
|
|
|
|
"# *# #@ #"
|
|
|
|
"# # *#**#* # #"
|
|
|
|
"# #*##*# * #"
|
|
|
|
"# *# *# #"
|
|
|
|
"# *#* *# #"
|
|
|
|
"# # # ##*# # #"
|
|
|
|
"# * #* H #"
|
|
|
|
"################",
|
|
|
|
|
|
|
|
"################"
|
|
|
|
"# * * H# #"
|
|
|
|
"# #*#* #* #"
|
|
|
|
"# #*# #* #"
|
|
|
|
"# * # *#* #"
|
|
|
|
"# #*# # *# #"
|
|
|
|
"# *#* # * # #"
|
|
|
|
"#*#@ # # #"
|
|
|
|
"################",
|
|
|
|
|
|
|
|
"################"
|
|
|
|
"#* ## ##*#"
|
|
|
|
"# # # #* #"
|
|
|
|
"# *# *## * #"
|
|
|
|
"# # #* #"
|
|
|
|
"# # *# #"
|
|
|
|
"# ## *# ##* #H#"
|
|
|
|
"# *# #* ##@#"
|
|
|
|
"################",
|
|
|
|
|
|
|
|
"################"
|
|
|
|
"# *#* #"
|
|
|
|
"##* *# ##* ##"
|
|
|
|
"# # # * # # #"
|
|
|
|
"# H # ## # @*#"
|
|
|
|
"# # # * # # #"
|
|
|
|
"## *## #* *##"
|
|
|
|
"# *#* #"
|
|
|
|
"################",
|
|
|
|
|
|
|
|
"################"
|
|
|
|
"# ### * ##"
|
|
|
|
"# # # ##"
|
|
|
|
"# ##* * #"
|
|
|
|
"# * * * ##"
|
|
|
|
"# # ###* #"
|
|
|
|
"# * * @ H * **#"
|
|
|
|
"################"
|
|
|
|
" ",
|
|
|
|
|
|
|
|
"################"
|
|
|
|
"#*# #*# #* # #"
|
|
|
|
"# # #"
|
|
|
|
"#* # #* * #"
|
|
|
|
"## #* * ###"
|
|
|
|
"# * # ###* #"
|
|
|
|
"# #@#H * #"
|
|
|
|
"################"
|
|
|
|
" ",
|
|
|
|
|
|
|
|
" ############## "
|
|
|
|
"# # #*# #* # #"
|
|
|
|
"# * # #"
|
|
|
|
"## # * #* #"
|
|
|
|
"# #* # ** * #"
|
|
|
|
"##* # ## * #"
|
|
|
|
"# #@#H * #"
|
|
|
|
" ############## "
|
|
|
|
" ",
|
|
|
|
|
|
|
|
"################"
|
|
|
|
"# # ##"
|
|
|
|
"# ##* * ##*##"
|
|
|
|
"# #* *# ###"
|
|
|
|
"# ** *# ## #"
|
|
|
|
"# #* * # ## #"
|
|
|
|
"# ## @#H###**#"
|
|
|
|
"################"
|
|
|
|
" ",
|
|
|
|
|
|
|
|
"################"
|
|
|
|
"# # #"
|
|
|
|
"# * ##* * #"
|
|
|
|
"# #* * ## #"
|
|
|
|
"# * ## #* #"
|
|
|
|
"# #* *# * #"
|
|
|
|
"# ##* #@ H #"
|
|
|
|
"################"
|
|
|
|
" " };
|
2011-06-30 15:17:21 +02:00
|
|
|
|
2011-07-01 11:55:06 +02:00
|
|
|
struct state {
|
2011-06-30 16:15:03 +02:00
|
|
|
char level[ LEVEL_HEIGHT * LEVEL_WIDTH ];
|
2011-06-30 17:53:59 +02:00
|
|
|
char moving;
|
|
|
|
int moves;
|
2011-07-01 11:55:06 +02:00
|
|
|
};
|
2011-06-30 16:15:03 +02:00
|
|
|
|
2011-07-01 11:55:06 +02:00
|
|
|
int count_gifts( struct state *s )
|
2011-06-30 15:17:21 +02:00
|
|
|
{
|
2011-06-30 16:15:03 +02:00
|
|
|
int i, n = 0;
|
2011-06-30 15:17:21 +02:00
|
|
|
for( i = 0 ; i < LEVEL_HEIGHT * LEVEL_WIDTH ; i++ ) {
|
2011-06-30 17:53:59 +02:00
|
|
|
if ( s->level[ i ] == GIFT ) {
|
2011-06-30 16:15:03 +02:00
|
|
|
n++;
|
2011-06-30 15:17:21 +02:00
|
|
|
}
|
2011-06-30 16:15:03 +02:00
|
|
|
}
|
|
|
|
return n;
|
|
|
|
}
|
|
|
|
|
2011-07-01 11:55:06 +02:00
|
|
|
void get_pos( struct state *s, int* pos )
|
2011-06-30 16:15:03 +02:00
|
|
|
{
|
|
|
|
int p;
|
|
|
|
|
2011-06-30 17:53:59 +02:00
|
|
|
p = (int)( strchr( s->level, s->moving ) - s->level );
|
2011-06-30 16:15:03 +02:00
|
|
|
|
|
|
|
pos[ 1 ] = p / LEVEL_WIDTH;
|
|
|
|
pos[ 0 ] = p - ( pos[ 1 ] * LEVEL_WIDTH );
|
|
|
|
}
|
|
|
|
|
2011-07-01 11:55:06 +02:00
|
|
|
char get_cell( struct state *s, int x, int y )
|
2011-06-30 16:15:03 +02:00
|
|
|
{
|
2011-06-30 17:53:59 +02:00
|
|
|
return s->level[ y * LEVEL_WIDTH + x ];
|
2011-06-30 16:15:03 +02:00
|
|
|
}
|
|
|
|
|
2011-07-01 11:55:06 +02:00
|
|
|
void set_cell( struct state *s, int x, int y, char item )
|
2011-06-30 16:15:03 +02:00
|
|
|
{
|
2011-06-30 17:53:59 +02:00
|
|
|
s->level[ y * LEVEL_WIDTH + x ] = item;
|
2011-06-30 16:15:03 +02:00
|
|
|
}
|
|
|
|
|
2011-07-01 11:55:06 +02:00
|
|
|
void load_level( struct state *s, char* lvl )
|
2011-06-30 17:53:59 +02:00
|
|
|
{
|
|
|
|
strncpy( s->level, lvl, LEVEL_HEIGHT * LEVEL_WIDTH );
|
|
|
|
s->moving = BALL;
|
|
|
|
s->moves = 0;
|
|
|
|
}
|
|
|
|
|
2011-07-01 11:55:06 +02:00
|
|
|
void switch_actor( struct state *s )
|
2011-06-30 17:53:59 +02:00
|
|
|
{
|
|
|
|
s->moving = (s->moving == BALL) ? CUBE : BALL;
|
|
|
|
}
|
|
|
|
|
2011-07-01 11:55:06 +02:00
|
|
|
int won_or_not( struct state *s )
|
2011-06-30 17:53:59 +02:00
|
|
|
{
|
|
|
|
return( count_gifts( s ) == 0 );
|
|
|
|
}
|
|
|
|
|
2011-07-01 12:27:57 +02:00
|
|
|
void move( struct state *s, direction where )
|
2011-06-30 17:53:59 +02:00
|
|
|
{
|
|
|
|
int dx = 0, dy = 0, tmpx, tmpy, *item_coord;
|
|
|
|
item_coord = malloc( sizeof( int ) * 2 );
|
|
|
|
get_pos( s, item_coord );
|
|
|
|
tmpx = item_coord[ 0 ];
|
|
|
|
tmpy = item_coord[ 1 ];
|
|
|
|
|
2011-07-01 12:27:57 +02:00
|
|
|
switch( where ) {
|
|
|
|
case UP:
|
2011-06-30 17:53:59 +02:00
|
|
|
dy--;
|
|
|
|
break;
|
2011-07-01 12:27:57 +02:00
|
|
|
case DOWN:
|
2011-06-30 17:53:59 +02:00
|
|
|
dy++;
|
|
|
|
break;
|
2011-07-01 12:27:57 +02:00
|
|
|
case LEFT:
|
2011-06-30 17:53:59 +02:00
|
|
|
dx--;
|
|
|
|
break;
|
2011-07-01 12:27:57 +02:00
|
|
|
case RIGHT:
|
2011-06-30 17:53:59 +02:00
|
|
|
dx++;
|
|
|
|
break;
|
|
|
|
default: break;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Calculating arrival coordinates */
|
2011-07-01 11:53:51 +02:00
|
|
|
while ( /* Hairy conditions ahead */
|
|
|
|
/* target cell is within level's boundaries */
|
2011-06-30 17:53:59 +02:00
|
|
|
( ( tmpx + dx >= 0 ) && ( tmpx + dx < LEVEL_WIDTH ) ) &&
|
|
|
|
( ( tmpy + dy >= 0 ) && ( tmpy + dy < LEVEL_HEIGHT ) ) &&
|
2011-07-01 11:53:51 +02:00
|
|
|
/* and target cell is empty */
|
2011-06-30 17:53:59 +02:00
|
|
|
( get_cell( s, tmpx + dx, tmpy + dy ) == VOID )
|
2011-07-01 11:53:51 +02:00
|
|
|
/* or, in case the ball is moving, target cell can be a gift (which we'll eat) */
|
2011-06-30 17:53:59 +02:00
|
|
|
|| ( s->moving == BALL && ( get_cell( s, tmpx + dx, tmpy + dy ) == GIFT ) )
|
|
|
|
) {
|
|
|
|
tmpx += dx;
|
|
|
|
tmpy += dy;
|
|
|
|
|
|
|
|
if ( s->moving == BALL && get_cell( s, tmpx, tmpy ) == GIFT ) {
|
|
|
|
set_cell( s, tmpx, tmpy, VOID );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Moving to arrival coordinates */
|
|
|
|
set_cell( s, item_coord[ 0 ], item_coord[ 1 ], VOID );
|
|
|
|
set_cell( s, tmpx, tmpy, s->moving );
|
|
|
|
|
|
|
|
free( item_coord );
|
|
|
|
}
|
|
|
|
|
2011-07-01 11:55:06 +02:00
|
|
|
void display_level( struct state *s )
|
2011-06-30 16:15:03 +02:00
|
|
|
{
|
|
|
|
int i, j, *ball, *cube;
|
|
|
|
|
2011-06-30 17:53:59 +02:00
|
|
|
printf( "%i gifts left", count_gifts( s ) );
|
|
|
|
if ( won_or_not( s ) ) {
|
|
|
|
printf( ", You WON !\n");
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
printf( ", go on.\n");
|
|
|
|
}
|
2011-06-30 16:15:03 +02:00
|
|
|
|
|
|
|
for( i = 0 ; i < LEVEL_HEIGHT ; i++ ) {
|
|
|
|
for( j = 0 ; j < LEVEL_WIDTH ; j++ ) {
|
2011-06-30 17:53:59 +02:00
|
|
|
switch( get_cell( s, j, i ) ) {
|
2011-06-30 16:15:03 +02:00
|
|
|
case WALL:
|
|
|
|
printf( "##" );
|
|
|
|
break;
|
|
|
|
case VOID:
|
|
|
|
printf( " " );
|
|
|
|
break;
|
|
|
|
case BALL:
|
|
|
|
printf( "()" );
|
|
|
|
break;
|
|
|
|
case CUBE:
|
|
|
|
printf( "[]" );
|
|
|
|
break;
|
|
|
|
case GIFT:
|
|
|
|
printf( "<>" );
|
|
|
|
break;
|
|
|
|
default: break;
|
|
|
|
}
|
2011-06-30 15:17:21 +02:00
|
|
|
}
|
2011-06-30 16:15:03 +02:00
|
|
|
printf( "\n" );
|
2011-06-30 15:17:21 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int main( int argc, char* argv[] )
|
|
|
|
{
|
2011-06-30 17:53:59 +02:00
|
|
|
int i = 0;
|
2011-07-01 11:55:06 +02:00
|
|
|
struct state *s = malloc( sizeof( struct state ) );
|
2011-06-30 17:53:59 +02:00
|
|
|
|
|
|
|
load_level( s, levels[ 0 ] );
|
2011-06-30 16:15:03 +02:00
|
|
|
|
2011-06-30 17:53:59 +02:00
|
|
|
display_level( s );
|
2011-07-01 11:53:51 +02:00
|
|
|
|
2011-06-30 17:53:59 +02:00
|
|
|
char* moves = "drdluruldrdlrulurudlurdul";
|
|
|
|
do {
|
|
|
|
fprintf( stderr, "move %c\n", moves[ i ]);
|
|
|
|
move( s, moves[ i ] );
|
|
|
|
display_level( s );
|
|
|
|
i++;
|
|
|
|
} while( ( ! won_or_not( s ) ) && ( moves[ i ] != '\0' ) );
|
|
|
|
display_level( s );
|
|
|
|
|
2011-07-01 11:48:08 +02:00
|
|
|
free( s );
|
2011-06-30 15:17:21 +02:00
|
|
|
return 0;
|
|
|
|
}
|