c-urs_-toil-s/star.c

540 lines
15 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
#include <ncurses.h>
2011-07-01 15:57:33 +02:00
char *levels[] = { "################"
"#@## x#H#"
"# x ###"
"# ##x #"
"# ## x ##"
"## x x x #"
"# x x## x #"
"# ##x x#"
2011-07-01 15:36:55 +02:00
"################",
" # # # # # ##"
"# x @#"
" #x #x x "
"# # x x # #"
" # x # "
"# #H# x #"
" # # # #xx#"
2011-07-01 15:36:55 +02:00
"# # "
" # # # ",
"################"
"# x#@#"
2011-07-01 15:36:55 +02:00
"# ## ##H#"
"# #x x #"
"# x x## x#"
"# #x x x# x##"
"# ##x #x x x###"
"#x ##x #"
2011-07-01 15:36:55 +02:00
"################",
"################"
"# #H#"
"# # #"
"##x#x x#x#x#x#x#"
"# # #x x# # # ##"
"##x#x#x x#x#x#x#"
2011-07-01 15:36:55 +02:00
"# # #"
"# # #@ #"
"################",
" ############## "
"#@ # # # #"
"# #x # x x # #"
2011-07-01 15:36:55 +02:00
"## # # #"
"#x #x# ##"
"## # x # #"
"#x# # # # #H#"
"# # x# #x#"
2011-07-01 15:36:55 +02:00
" ############## ",
" ############"
" # x #x x#"
" # x # ##"
" # x #"
"#@ x #"
"## x # ##"
"# x # #"
"#H # x ##x #"
2011-07-01 15:36:55 +02:00
"################",
"################"
"# #"
" ## ### #x ##x#"
" #x #x # # # # "
2011-07-01 15:36:55 +02:00
" # # ### ## "
" ## # #x# #x# "
2011-07-01 15:36:55 +02:00
"# #"
"# @#x H #x#"
2011-07-01 15:36:55 +02:00
"################",
"############### "
"# x## ##"
"# #x ## x #"
"# x## # #x #"
"## ## #x# #"
"## # x#x #"
"#xHx# x #@# #"
2011-07-01 15:36:55 +02:00
"## #"
" ###############",
" # ########### "
" #x#x # @#"
"#x x# x # "
" # # x## x# #"
"# #x #xHx x#"
"# x## # "
"#x#x # "
2011-07-01 15:36:55 +02:00
"# # "
"############ ",
" ########### "
"#### x #"
"# H ###x x# x#"
"# x #x #x # "
"# # x # x#"
"#x#x # x# #@# "
" #x ### ### "
2011-07-01 15:36:55 +02:00
"# # # # "
" ######### # #",
"################"
"# # @#"
"# #xx xx ##"
"## x ## x#"
"#x #x#xx ###"
2011-07-01 15:36:55 +02:00
"## ## ## #"
"#x x# x H x#"
"##x### # ##"
2011-07-01 15:36:55 +02:00
" ## ########### ",
"## ## #### "
"#@#####x ### x##"
"# xx x #"
"# ## ##x #x# #"
"# # x ###x ## #"
2011-07-01 15:36:55 +02:00
"# ## ## #H# #"
"# x #"
"# x #"
2011-07-01 15:36:55 +02:00
"################",
" ############## "
"# @# x ##"
"# # #x x## #"
"# x # #"
"# x #x#"
"# # x #"
"## x x #x#"
"#H # x # # #"
2011-07-01 15:36:55 +02:00
" ############## ",
"################"
"#x#x x#x#"
"# x#@ ## #"
"# H x #"
"# x# #"
"# x #"
"# x# # #"
"#x#x x#x#"
2011-07-01 15:36:55 +02:00
"################",
" ###### ####### "
"# x# x #"
"# # x # # x #"
"# @# #xx #x #"
" # # # x H# #"
"#x # #x #"
" # x # "
"#x x#"
2011-07-01 15:36:55 +02:00
" ############## ",
"################"
"## H#x x x#"
"#x @x#x ##"
"## ### x ##"
"## x#x# #"
"#xx x#x #"
"## x ####x #"
"##x# # #"
2011-07-01 15:36:55 +02:00
"################",
"################"
"# x# #@ #"
"# # x#xx#x # #"
"# #x##x# x #"
"# x# x# #"
"# x#x x# #"
"# # # ##x# # #"
"# x #x H #"
2011-07-01 15:36:55 +02:00
"################",
"################"
"# x x H# #"
"# #x#x #x #"
"# #x# #x #"
"# x # x#x #"
"# #x# # x# #"
"# x#x # x # #"
"#x#@ # # #"
2011-07-01 15:36:55 +02:00
"################",
"################"
"#x ## ##x#"
"# # # #x #"
"# x# x## x #"
"# # #x #"
"# # x# #"
"# ## x# ##x #H#"
"# x# #x ##@#"
2011-07-01 15:36:55 +02:00
"################",
"################"
"# x#x #"
"##x x# ##x ##"
"# # # x # # #"
"# H # ## # @x#"
"# # # x # # #"
"## x## #x x##"
"# x#x #"
2011-07-01 15:36:55 +02:00
"################",
"################"
"# ### x ##"
2011-07-01 15:36:55 +02:00
"# # # ##"
"# ##x x #"
"# x x x ##"
"# # ###x #"
"# x x @ H x xx#"
2011-07-01 15:36:55 +02:00
"################"
" ",
"################"
"#x# #x# #x # #"
2011-07-01 15:36:55 +02:00
"# # #"
"#x # #x x #"
"## #x x ###"
"# x # ###x #"
"# #@#H x #"
2011-07-01 15:36:55 +02:00
"################"
" ",
" ############## "
"# # #x# #x # #"
"# x # #"
"## # x #x #"
"# #x # xx x #"
"##x # ## x #"
"# #@#H x #"
2011-07-01 15:36:55 +02:00
" ############## "
" ",
"################"
"# # ##"
"# ##x x ##x##"
"# #x x# ###"
"# xx x# ## #"
"# #x x # ## #"
"# ## @#H###xx#"
2011-07-01 15:36:55 +02:00
"################"
" ",
"################"
"# # #"
"# x ##x x #"
"# #x x ## #"
"# x ## #x #"
"# #x x# x #"
"# ##x #@ H #"
2011-07-01 15:36:55 +02:00
"################"
" " };
2011-06-30 15:17:21 +02:00
2011-07-01 17:31:01 +02:00
/* levels have fixed, hardcoded dimensions */
#define LEVEL_HEIGHT 9
#define LEVEL_WIDTH 16
enum {
color_BALL = 1,
color_CUBE,
color_VOID,
color_GIFT,
color_WALL,
color_BALL_SELECTED,
color_CUBE_SELECTED,
};
typedef enum {
WALL ='#',
BALL ='@',
CUBE ='H',
VOID =' ',
GIFT ='x'
} cell;
typedef enum {
UP ='u',
DOWN ='d',
LEFT ='l',
RIGHT ='r'
} direction;
struct state {
char level[ LEVEL_HEIGHT * LEVEL_WIDTH ];
char moving;
int moves;
};
int count_gifts( struct state *s )
2011-06-30 16:15:03 +02:00
{
2011-07-01 17:31:01 +02:00
int i, n = 0;
for( i = 0 ; i < LEVEL_HEIGHT * LEVEL_WIDTH ; i++ ) {
if ( s->level[ i ] == GIFT ) {
n++;
}
}
return n;
}
void get_pos( struct state *s, int* pos )
{
int p;
p = (int)( strchr( s->level, s->moving ) - s->level );
pos[ 1 ] = p / LEVEL_WIDTH;
pos[ 0 ] = p - ( pos[ 1 ] * LEVEL_WIDTH );
}
2011-07-01 15:36:55 +02:00
2011-07-01 17:31:01 +02:00
char get_cell( struct state *s, int x, int y )
{
return s->level[ y * LEVEL_WIDTH + x ];
}
void set_cell( struct state *s, int x, int y, cell value )
{
s->level[ y * LEVEL_WIDTH + x ] = value;
}
void load_level( struct state *s, char* lvl )
{
strncpy( s->level, lvl, LEVEL_HEIGHT * LEVEL_WIDTH );
s->moving = BALL;
s->moves = 0;
}
void switch_actor( struct state *s )
{
s->moving = (s->moving == BALL) ? CUBE : BALL;
}
int won_or_not( struct state *s )
{
return( count_gifts( s ) == 0 );
}
void make_a_move( struct state *s, direction where )
{
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 ];
switch( where ) {
case UP:
dy--;
break;
case DOWN:
dy++;
break;
case LEFT:
dx--;
break;
case RIGHT:
dx++;
break;
default: break;
2011-06-30 17:53:59 +02:00
}
2011-07-01 17:31:01 +02:00
/* Calculating arrival coordinates */
while ( /* Hairy conditions ahead */
/* target cell is within level's boundaries */
( ( tmpx + dx >= 0 ) && ( tmpx + dx < LEVEL_WIDTH ) ) &&
( ( tmpy + dy >= 0 ) && ( tmpy + dy < LEVEL_HEIGHT ) ) &&
/* and target cell is empty */
( 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) */
|| ( 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 );
}
2011-06-30 17:53:59 +02:00
}
2011-07-01 15:36:55 +02:00
2011-07-01 17:31:01 +02:00
/* Moving to arrival coordinates */
set_cell( s, item_coord[ 0 ], item_coord[ 1 ], VOID );
set_cell( s, tmpx, tmpy, s->moving );
s->moves++;
free( item_coord );
}
void display_level( struct state *s )
{
int i, j, *ball, *cube;
2011-06-30 16:15:03 +02:00
for( i = 0 ; i < LEVEL_HEIGHT ; i++ ) {
2011-07-01 15:36:55 +02:00
for( j = 0 ; j < LEVEL_WIDTH ; j++ ) {
switch( get_cell( s, j, i ) ) {
case WALL:
2011-07-01 17:22:14 +02:00
attron( COLOR_PAIR( color_WALL ));
mvprintw( i+1, j*2, "##" );
2011-07-01 17:22:14 +02:00
attroff( COLOR_PAIR( color_WALL ));
2011-07-01 15:36:55 +02:00
break;
case VOID:
2011-07-01 17:22:14 +02:00
attron( COLOR_PAIR( color_VOID ));
mvprintw( i+1, j*2, " " );
2011-07-01 17:22:14 +02:00
attroff( COLOR_PAIR( color_VOID ));
2011-07-01 15:36:55 +02:00
break;
case BALL:
2011-07-01 17:10:22 +02:00
if ( s->moving == BALL ) {
attron( A_BOLD );
2011-07-01 17:22:14 +02:00
attron( COLOR_PAIR( color_BALL_SELECTED ));
}
else {
attron( COLOR_PAIR( color_BALL ));
2011-07-01 17:10:22 +02:00
}
mvprintw( i+1, j*2, "()" );
2011-07-01 17:10:22 +02:00
if ( s->moving == BALL ) {
attroff( A_BOLD );
}
2011-07-01 15:36:55 +02:00
break;
case CUBE:
2011-07-01 17:10:22 +02:00
if ( s->moving == CUBE ) {
attron( A_BOLD );
2011-07-01 17:22:14 +02:00
attron( COLOR_PAIR( color_BALL_SELECTED ));
}
else {
attron( COLOR_PAIR( color_BALL ));
2011-07-01 17:10:22 +02:00
}
mvprintw( i+1, j*2, "[]" );
2011-07-01 17:10:22 +02:00
if ( s->moving == CUBE ) {
attroff( A_BOLD );
2011-07-01 17:22:14 +02:00
attroff( COLOR_PAIR( color_CUBE_SELECTED ));
}
else {
attroff( COLOR_PAIR( color_CUBE ));
2011-07-01 17:10:22 +02:00
}
2011-07-01 15:36:55 +02:00
break;
case GIFT:
2011-07-01 17:22:14 +02:00
attron( COLOR_PAIR( color_GIFT ));
mvprintw( i+1, j*2, "<>" );
2011-07-01 17:22:14 +02:00
attroff( COLOR_PAIR( color_GIFT ));
2011-07-01 15:36:55 +02:00
break;
default: break; /* ignore newlines */
}
}
2011-06-30 15:17:21 +02:00
}
2011-07-01 17:31:01 +02:00
mvprintw( 1, 35, "%i gifts left", count_gifts( s ) );
mvprintw( 2, 35, "%i moves made", s->moves );
refresh();
2011-06-30 15:17:21 +02:00
}
int main( int argc, char* argv[] )
{
int i = 0, lvl = 0, key, load_requested = 0;
2011-07-01 11:55:06 +02:00
struct state *s = malloc( sizeof( struct state ) );
2011-07-01 15:36:55 +02:00
2011-07-01 16:57:19 +02:00
/* trick to count how many levels we have */
int nb_levels = sizeof( levels ) / sizeof( levels[ 0 ] );
/* ncurses */
2011-07-01 17:35:03 +02:00
WINDOW *w_main = initscr( ); /* why can't stdscr be used transparently? */
cbreak();
noecho();
nonl();
intrflush( w_main, FALSE );
keypad( w_main, TRUE );
2011-07-01 17:24:03 +02:00
if ( has_colors( ) == TRUE ) {
start_color( );
init_pair( color_CUBE, COLOR_RED, COLOR_BLACK );
init_pair( color_BALL, COLOR_BLUE, COLOR_BLACK );
init_pair( color_GIFT, COLOR_YELLOW, COLOR_BLACK );
init_pair( color_WALL, COLOR_WHITE, COLOR_WHITE );
init_pair( color_VOID, COLOR_BLACK, COLOR_BLACK );
init_pair( color_CUBE_SELECTED, COLOR_RED, COLOR_YELLOW );
init_pair( color_BALL_SELECTED, COLOR_BLUE, COLOR_YELLOW );
}
/* load the first level to start the loop in a correct state */
load_level( s, levels[ lvl ] );
2011-06-30 17:53:59 +02:00
do {
if ( won_or_not( s ) ) {
lvl++;
load_requested = 1;
}
if ( load_requested == 1 ) {
load_requested = 0;
load_level( s, levels[ lvl ] );
}
2011-07-01 15:36:55 +02:00
display_level( s );
key = getch();
switch( key ) {
case KEY_UP:
make_a_move( s, UP );
break;
case KEY_DOWN:
make_a_move( s, DOWN );
break;
case KEY_LEFT:
make_a_move( s, LEFT );
break;
case KEY_RIGHT:
make_a_move( s, RIGHT );
break;
case ' ':
switch_actor( s );
2011-07-01 16:58:48 +02:00
break;
case 'n':
if ( lvl < nb_levels - 1 ) {
lvl++;
load_requested = 1;
}
break;
case 'p':
if ( lvl > 0 ) {
lvl--;
load_requested = 1;
}
break;
case 'r':
load_requested = 1;
break;
default:
break;
}
2011-07-01 16:57:19 +02:00
} while( lvl < nb_levels && (( key != 'q' ) && ( key != 'Q' )) );
2011-06-30 17:53:59 +02:00
display_level( s );
2011-07-01 11:48:08 +02:00
free( s );
echo();
nocbreak();
endwin();
2011-06-30 15:17:21 +02:00
return 0;
}