c-urs_-toil-s/star.c

423 lines
9.3 KiB
C
Raw Normal View History

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 */
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 ) ) &&
/* and target cell is empty */
2011-06-30 17:53:59 +02:00
( get_cell( s, tmpx + dx, tmpy + dy ) == VOID )
/* 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-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;
}