X-Git-Url: http://royale.zerezo.com/git/?p=zeRace;a=blobdiff_plain;f=zeRace.c;h=9277dd3b74bf5935dd74b00a43745bec15f3afbf;hp=ce90ba1c1eeb15bda14f16d0ef6ae0fa20cef447;hb=HEAD;hpb=866e35b81c26169886388f7c93dce52f2a42809f diff --git a/zeRace.c b/zeRace.c index ce90ba1..9277dd3 100644 --- a/zeRace.c +++ b/zeRace.c @@ -1,5 +1,5 @@ /* - * zeRace 0.2, a funny retro racing game + * zeRace 0.7, a funny retro racing game * http://royale.zerezo.com/zerace/ * * Copyright (C) 2004 Antoine Jacquet @@ -30,30 +30,29 @@ #include #include #include "sdl.h" +#include "car.h" +#include "tracklist.h" +#include "network.h" +#include /* configuration constants */ #define COEFF 1 #define DELAY 7 -#define MAXLINELENGTH 80 -#define VERSION "0.2" #define WIDTH 1024 #define HEIGHT 768 +#define MAXRECORDKEYS 9999 +#define NB_CARS 16 -/* tracklist : double chained list */ -struct _tracklist -{ - char *name; - char *title; - char *author; - char *version; - char *full; - char *function; - int x,y,a; - int best_time; - char *best_pseudo; - struct _tracklist *prev; - struct _tracklist *next; -} *tracklist=NULL; +/* some usefull colors */ +#define C_BLACK SDL_MapRGB(screen->format,0,0,0) +#define C_WHITE SDL_MapRGB(screen->format,255,255,255) +#define C_RED SDL_MapRGB(screen->format,255,0,0) +#define C_ORANGE SDL_MapRGB(screen->format,255,200,0) +#define C_YELLOW SDL_MapRGB(screen->format,255,255,100) +#define T_BLACK SDL_MapRGB(cir->format,0,0,0) + +/* tracklist */ +struct _tracklist *tracklist; /* user setup */ struct _config @@ -67,19 +66,35 @@ struct _config SDLKey down; SDLKey left; SDLKey right; - char color; -} config = {"anonymous","",0,0,1,SDLK_UP,SDLK_DOWN,SDLK_LEFT,SDLK_RIGHT,6}; + int color; + SDLKey boss; + int bynight; + int internet; +} config = {"anonymous","",0,0,1,SDLK_UP,SDLK_DOWN,SDLK_LEFT,SDLK_RIGHT,6,SDLK_b,0,1}; + +/* full script for a lap */ +struct _record +{ + float x,y,angle,speed; + char keys[MAXRECORDKEYS]; + int time; +}; /* display and all directions for the car */ SDL_Surface *screen; -SDL_Surface *cars[256]; +SDL_Surface *cars[NB_CARS][256]; + +/* network stuff */ +UDPsocket udpsock=NULL; +UDPpacket *packet; +int network_speed=1; /* read the user configuration file */ void zeRace_read_config() { FILE *fic; - if ((fic=fopen("zeRace.cfg","rt"))==NULL) + if ((fic=fopen("zeRace.cfg","rb"))==NULL) { fprintf(stderr,"can't open config file \"zeRace.cfg\"\n"); return; @@ -93,7 +108,7 @@ void zeRace_read_config() void zeRace_save_config() { FILE *fic; - if ((fic=fopen("zeRace.cfg","wt"))==NULL) + if ((fic=fopen("zeRace.cfg","wb"))==NULL) { fprintf(stderr,"can't create config file \"zeRace.cfg\"\n"); return; @@ -115,64 +130,6 @@ void zeRace_exit() } -/* get available local tracks */ -void zeRace_get_tracks() -{ - FILE *fic; - char line[MAXLINELENGTH]; - struct _tracklist *tmp=NULL,*first=NULL; - - if ((fic=fopen("tracks/list.txt","rt"))==NULL) - { - fprintf(stderr,"can't open track list\n"); - zeRace_exit(); - } - while (!feof(fic)) - { - tmp=(struct _tracklist *)malloc(sizeof(struct _tracklist)); - if (first==NULL) first=tmp; - fgets(line,MAXLINELENGTH,fic); - tmp->name=(char *)malloc(strlen(line)+1); - strcpy(tmp->name,line); - tmp->name[strlen(tmp->name)-1]='\0'; - fgets(line,MAXLINELENGTH,fic); - tmp->title=(char *)malloc(strlen(line)+1); - strcpy(tmp->title,line); - fgets(line,MAXLINELENGTH,fic); - tmp->author=(char *)malloc(strlen(line)+1); - strcpy(tmp->author,line); - fgets(line,MAXLINELENGTH,fic); - tmp->version=(char *)malloc(strlen(line)+1); - strcpy(tmp->version,line); - fgets(line,MAXLINELENGTH,fic); - tmp->x=atoi(line); - fgets(line,MAXLINELENGTH,fic); - tmp->y=atoi(line); - fgets(line,MAXLINELENGTH,fic); - tmp->a=atoi(line); - fgets(line,MAXLINELENGTH,fic); - tmp->best_time=atoi(line); - fgets(line,MAXLINELENGTH,fic); - tmp->best_pseudo=(char *)malloc(strlen(line)+1); - strcpy(tmp->best_pseudo,line); - tmp->full=(char *)malloc(strlen(line)+20); - sprintf(tmp->full,"tracks/%s.png",tmp->name); - tmp->function=(char *)malloc(strlen(line)+30); - sprintf(tmp->function,"tracks/%s_function.png",tmp->name); - tmp->prev=tracklist; - if (tmp->prev) tmp->prev->next=tmp; - tracklist=tmp; - /* skip one line */ - fgets(line,MAXLINELENGTH,fic); - } - fclose(fic); - if (!tmp) { fprintf(stderr,"no circuits found !\n"); zeRace_exit(); } - while (tmp->prev) tmp=tmp->prev; - tmp->prev=tracklist; - tracklist->next=tmp; -} - - /* check for a newer version online to warn the user */ void zeRace_check_version() { @@ -186,17 +143,19 @@ void zeRace_check_version() char response[1024],*tmp,*version; int len,result; + if (!config.internet) return; + printf("checking version... "); fflush(stdout); - if(SDLNet_ResolveHost(&ip,"royale.zerezo.com",80)==-1) + if (SDLNet_ResolveHost(&ip,"royale.zerezo.com",80)==-1) { fprintf(stderr,"SDLNet_ResolveHost: %s\n",SDLNet_GetError()); return; } tcpsock=SDLNet_TCP_Open(&ip); - if(!tcpsock) + if (!tcpsock) { fprintf(stderr,"SDLNet_TCP_Open: %s\n",SDLNet_GetError()); return; @@ -204,7 +163,7 @@ void zeRace_check_version() len=strlen(request); result=SDLNet_TCP_Send(tcpsock,request,len); - if(resultw;x++) for (y=0;yh;y++) - { - int x2,y2; - x2=(x-cars[i]->w/2.0)*tcos+(y-cars[i]->h/2.0)*tsin+car->w/2.0; - y2=(x-cars[i]->w/2.0)*tsin-(y-cars[i]->h/2.0)*tcos+car->h/2.0; - if (x2>0 && x2w && y2>0 && y2h) - putpixel(cars[i],x,y,getpixel(car,x2,y2)); - } - } - SDL_FreeSurface(car); + for (i=0;iw;x++) for (y=0;yh;y++) + { + int x2,y2; + x2=(x-cars[i][j]->w/2.0)*tcos+(y-cars[i][j]->h/2.0)*tsin+car->w/2.0; + y2=(x-cars[i][j]->w/2.0)*tsin-(y-cars[i][j]->h/2.0)*tcos+car->h/2.0; + if (x2>0 && x2w && y2>0 && y2h) + putpixel(cars[i][j],x,y,getpixel(car,x2,y2)); + } + } + SDL_FreeSurface(car); + } } /* initialize the game */ void zeRace_init() { - int flags,i; + int flags; + struct _tracklist *loopcheck; /* do a clean exit in case of emergency */ signal(SIGINT,zeRace_exit); signal(SIGTERM,zeRace_exit); + /* initialize SDL */ + if (SDL_Init(SDL_INIT_VIDEO|SDL_INIT_AUDIO)<0) + { + fprintf(stderr,"could not initialize SDL : %s\n",SDL_GetError()); + zeRace_exit(); + } + atexit(SDL_Quit); + + /* initialize SDLNet */ + if (SDLNet_Init()==-1) + { + fprintf(stderr,"could not initialize SDLNet : %s\n",SDLNet_GetError()); + zeRace_exit(); + } + /* read the user configuration file */ zeRace_read_config(); @@ -330,23 +377,28 @@ void zeRace_init() zeRace_update_tracks(); /* get the list of local tracks */ - zeRace_get_tracks(); + if (!zeRace_get_tracks(&tracklist)) zeRace_exit(); - srand(time(NULL)); - - if (SDL_Init(SDL_INIT_VIDEO|SDL_INIT_AUDIO)<0) + /* download missing files */ + loopcheck=tracklist; + do { - fprintf(stderr,"could not initialize SDL : %s\n",SDL_GetError()); - zeRace_exit(); - } - atexit(SDL_Quit); + zeRace_download_file(tracklist->full); + zeRace_download_file(tracklist->function); + tracklist=tracklist->next; + } while (tracklist!=loopcheck); + + srand(time(NULL)); - if(SDLNet_Init()==-1) + packet=SDLNet_AllocPacket(1024); + if (!packet) { - fprintf(stderr,"could not initialize SDLNet : %s\n",SDLNet_GetError()); + fprintf(stderr,"SDLNet_AllocPacket: %s\n",SDLNet_GetError()); zeRace_exit(); } + SDL_EnableKeyRepeat(SDL_DEFAULT_REPEAT_DELAY,SDL_DEFAULT_REPEAT_INTERVAL); + flags=SDL_HWSURFACE|SDL_ANYFORMAT; if (config.fullscreen) flags|=SDL_FULLSCREEN; @@ -366,18 +418,13 @@ void zeRace_init() zeRace_exit(); } - /* allocate memory for car sprites */ - for (i=0;i<256;i++) if ((cars[i]=SDL_CreateRGBSurface(SDL_SWSURFACE,30,30,32,0x000000ff,0x0000ff00,0x00ff0000,0xff000000))==NULL) - { - fprintf(stderr,"CreateRGBSurface failed: %s\n",SDL_GetError()); - zeRace_exit(); - }; + /* pre-calculate car sprites */ zeRace_generate_cars(); } /* send the best time for this race to the web server */ -void zeRace_send_time(float x,float y,float speed,float angle,int btime,char *bkeys) +void zeRace_send_time(struct _record *record) { IPaddress ip; TCPsocket tcpsock; @@ -400,28 +447,33 @@ void zeRace_send_time(float x,float y,float speed,float angle,int btime,char *bk char *msg9="&bkeys="; int len,result; + if (!config.internet) return; + + /* if the best time is small enought to save all keys, send it */ + if (record->time>=MAXRECORDKEYS) return; + printf("sending time... "); fflush(stdout); - if(SDLNet_ResolveHost(&ip,"royale.zerezo.com",80)==-1) + if (SDLNet_ResolveHost(&ip,"royale.zerezo.com",80)==-1) { fprintf(stderr,"SDLNet_ResolveHost: %s\n",SDLNet_GetError()); return; } tcpsock=SDLNet_TCP_Open(&ip); - if(!tcpsock) + if (!tcpsock) { fprintf(stderr,"SDLNet_TCP_Open: %s\n",SDLNet_GetError()); return; } - temp=(char *)malloc(strlen(msg1)+strlen(config.pseudo)+strlen(msg2)+strlen(config.url)+strlen(msg3)+strlen(tracklist->name)+strlen(msg4)+10+strlen(msg5)+10+strlen(msg6)+10+strlen(msg7)+10+strlen(msg8)+10+strlen(msg9)+strlen(bkeys)+100); - sprintf(temp,"%s%s%s%s%s%s%s%d%s%f%s%f%s%f%s%f%s%s\n",msg1,config.pseudo,msg2,config.url,msg3,tracklist->name,msg4,btime,msg5,x,msg6,y,msg7,speed,msg8,angle,msg9,bkeys); + temp=(char *)malloc(strlen(msg1)+strlen(config.pseudo)+strlen(msg2)+strlen(config.url)+strlen(msg3)+strlen(tracklist->name)+strlen(msg4)+10+strlen(msg5)+10+strlen(msg6)+10+strlen(msg7)+10+strlen(msg8)+10+strlen(msg9)+strlen(record->keys)+100); + sprintf(temp,"%s%s%s%s%s%s%s%d%s%f%s%f%s%f%s%f%s%s\n",msg1,config.pseudo,msg2,config.url,msg3,tracklist->name,msg4,record->time,msg5,record->x,msg6,record->y,msg7,record->speed,msg8,record->angle,msg9,record->keys); len=strlen(temp); result=SDLNet_TCP_Send(tcpsock,temp,len); - if(result1) + { + putpixel(screen,x-1,y,pixel); + putpixel(screen,x+1,y,pixel); + putpixel(screen,x,y-1,pixel); + putpixel(screen,x,y+1,pixel); + } + if (r>2) + { + putpixel(screen,x-2,y,pixel); + putpixel(screen,x+2,y,pixel); + putpixel(screen,x,y-2,pixel); + putpixel(screen,x,y+2,pixel); + putpixel(screen,x-1,y-1,pixel); + putpixel(screen,x-1,y+1,pixel); + putpixel(screen,x+1,y-1,pixel); + putpixel(screen,x+1,y+1,pixel); + } +} + + /* launch a new race */ -void zeRace_launch() +void zeRace_launch(int alltime,int go) { - SDL_Surface *cir,*fun; - SDL_Rect pos,size; + SDL_Surface *cir,*fun,*hilight; + SDL_Rect pos; SDL_Event event; - int ku=0,kd=0,kl=0,kr=0; - int i,time=0,lastcheck=0,btime=10000; - float ox,oy; - float x,y,angle,speed; - float sx,sy,sangle,sspeed; - float bx,by,bangle,bspeed; - int c,r,g,b; - char keys[10000]; - char bkeys[10000]; + int ku=0,kd=0,kl=0,kr=0,i; + struct _car car; + struct _record current,best; Mix_Music *light,*engine,*crash,*slide; - int lastsound_time=-999,alltime=0,lastsound=0; + int lastsound_time=-999,lastsound=0; + int delay=DELAY; + int lastack=alltime; + struct _record net; + struct _car oldnetpos[MAX_CLIENTS],newnetpos[MAX_CLIENTS]; + int l=80,o=15; + + /* free memory */ + void free_mem() + { + SDL_FreeSurface(cir); + SDL_FreeSurface(fun); + if (config.sound) + { + Mix_FreeMusic(light); + Mix_FreeMusic(engine); + Mix_FreeMusic(crash); + Mix_FreeMusic(slide); + } + return; + } cir=IMG_Load(tracklist->full); + /* dark the track if it is night */ + if (config.bynight) + { + for (pos.x=0;pos.xw;pos.x++) + for (pos.y=0;pos.yh;pos.y++) + { + Uint32 c; + Uint8 r,g,b; + c=getpixel(cir,pos.x,pos.y); + SDL_GetRGB(c,cir->format,&r,&g,&b); + r*=0.3; + g*=0.3; + b*=0.3; + putpixel(cir,pos.x,pos.y,SDL_MapRGB(cir->format,r,g,b)); + } + } fun=IMG_Load(tracklist->function); + hilight=IMG_Load("sprites/light.png"); - sspeed=speed=0; - sangle=angle=(tracklist->a*2*M_PI/360); - sx=x=tracklist->x; - sy=y=tracklist->y; - lastcheck=0; - time=0; + current.speed=car.speed=0; + current.angle=car.angle=tracklist->a*2*M_PI/360; + current.x=car.ox=car.x=tracklist->x; + current.y=car.oy=car.y=tracklist->y; + car.lastcheck=0; + car.lapflag=0; + car.w=cars[0][0]->w; + car.h=cars[0][0]->h; + current.time=0; + best.time=MAXRECORDKEYS; + net.time=0; + memset(oldnetpos,0,MAX_CLIENTS*sizeof(struct _car)); + memset(newnetpos,0,MAX_CLIENTS*sizeof(struct _car)); if (config.sound) if (!(light=Mix_LoadMUS("sounds/light.wav")) || !(engine=Mix_LoadMUS("sounds/engine.wav")) || !(crash=Mix_LoadMUS("sounds/crash.wav")) || !(slide=Mix_LoadMUS("sounds/slide.wav"))) { @@ -490,10 +605,10 @@ void zeRace_launch() char startup[15]="sprites/?.png"; SDL_Surface *temp; startup[8]='0'+i; - pos.x=x; - pos.y=y; + pos.x=car.x-car.w/2; + pos.y=car.y-car.h/2; SDL_BlitSurface(cir,NULL,screen,NULL); - SDL_BlitSurface(cars[(unsigned char)(256*angle/2.0/M_PI)%256],NULL,screen,&pos); + SDL_BlitSurface(cars[config.color][(unsigned char)(256*car.angle/2.0/M_PI)%256],NULL,screen,&pos); if (i!=4 && i!=-1) { temp=IMG_Load(startup); @@ -504,48 +619,316 @@ void zeRace_launch() } if (config.sound) if (i!=4) Mix_PlayMusic(light,1); SDL_Flip(screen); + if (!go) break; if (i!=-1) SDL_Delay(1000); } /* main loop */ for (;;) { + /* look for user interaction */ + while (SDL_PollEvent(&event)) + { + switch (event.type) + { + case SDL_QUIT: + zeRace_exit(); + break; + case SDL_KEYDOWN: + switch (event.key.keysym.sym) + { + case SDLK_ESCAPE: + if (udpsock) + { + print(screen,WIDTH/2-strlen("Disconnecting !")*5,HEIGHT/2-10,"Disconnecting !"); + strcpy(packet->data,"disconnect"); + packet->len=strlen(packet->data)+1; + SDLNet_UDP_Send(udpsock,-1,packet); + SDL_Flip(screen); + } + zeRace_send_time(&best); + free_mem(); + return; + default: + i=event.key.keysym.sym; + if (i==config.up) ku=1; + if (i==config.down) kd=1; + if (i==config.left) kl=1; + if (i==config.right) kr=1; + if (i==config.boss) + { + /* display the boss screen */ + SDL_Surface *boss; + boss=IMG_Load("sprites/boss.png"); + SDL_BlitSurface(boss,NULL,screen,NULL); + SDL_FreeSurface(boss); + SDL_Flip(screen); + /* and wait until the user press another key */ + for (;;) if (SDL_PollEvent(&event)) { if (event.type==SDL_KEYDOWN) break; } else SDL_Delay(10); + SDL_BlitSurface(cir,NULL,screen,NULL); + SDL_Flip(screen); + } + break; + } + break; + case SDL_KEYUP: + i=event.key.keysym.sym; + if (i==config.up) ku=0; + if (i==config.down) kd=0; + if (i==config.left) kl=0; + if (i==config.right) kr=0; + break; + } + } + + /* save pressed keys to validate best time */ + if (current.timedata,"positions")==0) + { + int servertime,clienttime,nb; + servertime=SDLNet_Read32(packet->data+strlen("positions")+1); + clienttime=SDLNet_Read32(packet->data+strlen("positions")+1+4); + nb=SDLNet_Read16(packet->data+strlen("positions")+1+4+4); + if (clienttime>lastack) + { + memcpy(net.keys,net.keys+clienttime-lastack,net.time+1); + net.time-=clienttime-lastack; + if (clienttime>servertime+5) delay+=DELAY; + if (clienttimedata+strlen("positions")+1+4+4+2+i*14); + newnetpos[i].y=SDLNet_Read16(packet->data+strlen("positions")+1+4+4+2+i*14+2); + newnetpos[i].angle=(float)SDLNet_Read16(packet->data+strlen("positions")+1+4+4+2+i*14+2+2)/1000; + newnetpos[i].color=SDLNet_Read16(packet->data+strlen("positions")+1+4+4+2+i*14+2+2+2); + newnetpos[i].lights_brake=SDLNet_Read16(packet->data+strlen("positions")+1+4+4+2+i*14+2+2+2+2); + newnetpos[i].lights_backwards=SDLNet_Read16(packet->data+strlen("positions")+1+4+4+2+i*14+2+2+2+2+2); + newnetpos[i].lights_warning=SDLNet_Read16(packet->data+strlen("positions")+1+4+4+2+i*14+2+2+2+2+2+2); + } + lastack=clienttime; + } + } + else if (strcmp(packet->data,"collision")==0) + { + net.time=-1; + net.keys[0]='\0'; + lastack=SDLNet_Read32(packet->data+strlen("collision")+1); + car.x=(float)SDLNet_Read32(packet->data+strlen("collision")+1+4)/65536-100; + car.y=(float)SDLNet_Read32(packet->data+strlen("collision")+1+4+4)/65536-100; + car.speed=(float)SDLNet_Read32(packet->data+strlen("collision")+1+4+4+4)/65536-100; + car.angle=(float)SDLNet_Read32(packet->data+strlen("collision")+1+4+4+4+4)/65536-100; + } + else /* end of this network race */ + { + zeRace_send_time(&best); + free_mem(); + return; + } + if (strlen(net.keys)!=0) + { + tmp=packet->data; + strcpy(tmp,"keys"); + tmp+=strlen(tmp)+1; + SDLNet_Write32(lastack,tmp); + tmp+=4; + SDLNet_Write32(car.x,tmp); + tmp+=4; + SDLNet_Write32(car.y,tmp); + tmp+=4; + strcpy(tmp,net.keys); + tmp+=strlen(tmp)+1; + packet->len=(void *)tmp-(void *)packet->data+10; + if (net.time%network_speed==0) if (!SDLNet_UDP_Send(udpsock,-1,packet)) + { + fprintf(stderr,"SDLNet_UDP_Send: %s\n",SDLNet_GetError()); + exit(2); + }; + } + } + + /* clear the old network position */ + if (udpsock) for (i=0;iw; - size.h=cars[0]->h; - pos.x=x; - pos.y=y; + pos.x=car.ox-car.w/2; + pos.y=car.oy-car.h/2; + pos.w=car.w; + pos.h=car.h; SDL_BlitSurface(cir,&pos,screen,&pos); - /* save the old position and compute the new one */ - ox=x; - oy=y; - speed*=0.995; - x=x-cos(angle)*speed; - y=y-sin(angle)*speed; + /* clear the lights */ + if (config.bynight) + { + pos.x=car.ox-cos(car.angle)*l-sin(car.angle)*o; + pos.y=car.oy-sin(car.angle)*l+cos(car.angle)*o; + pos.x-=60; + pos.y-=60; + pos.w=120; + pos.h=120; + SDL_BlitSurface(cir,&pos,screen,&pos); + pos.x=car.ox-cos(car.angle)*l+sin(car.angle)*o; + pos.y=car.oy-sin(car.angle)*l-cos(car.angle)*o; + pos.x-=60; + pos.y-=60; + pos.w=120; + pos.h=120; + SDL_BlitSurface(cir,&pos,screen,&pos); + } + + /* display the network car at the new position */ + if (udpsock) for (i=0;iscreen->w-cars[0]->w || y<0 || y>screen->h-cars[0]->h) + /* display the car at the new position */ + pos.x=car.x-car.w/2; + pos.y=car.y-car.h/2; + pos.w=car.w; + pos.h=car.h; + SDL_BlitSurface(cars[config.color][(unsigned char)(256*car.angle/2.0/M_PI)%256],NULL,screen,&pos); + + /* if the car is braking, display red lights */ + if (car.lights_brake) { - x=ox; - y=oy; - speed=0; + lights(car.x+cos(car.angle)*car.w/3-sin(car.angle)*4,car.y+sin(car.angle)*car.h/3+cos(car.angle)*4,3,C_RED); + lights(car.x+cos(car.angle)*car.w/3+sin(car.angle)*4,car.y+sin(car.angle)*car.h/3-cos(car.angle)*4,3,C_RED); } - /* display the car at the new position and update display */ - pos.x=x; - pos.y=y; - i=(unsigned char)(256*angle/2.0/M_PI)%256; - SDL_BlitSurface(cars[i],NULL,screen,&pos); - SDL_UpdateRect(screen,ox,oy,cars[i]->w,cars[i]->h); - SDL_UpdateRect(screen,x,y,cars[i]->w,cars[i]->h); + /* if the car is going backwards, display white lights */ + if (car.lights_backwards) + { + lights(car.x+cos(car.angle)*car.w/3-sin(car.angle)*4,car.y+sin(car.angle)*car.h/3+cos(car.angle)*4,3,C_WHITE); + lights(car.x+cos(car.angle)*car.w/3+sin(car.angle)*4,car.y+sin(car.angle)*car.h/3-cos(car.angle)*4,3,C_WHITE); + } + + /* if the car is stopped, then warning */ + if (car.lights_warning && alltime/75%2) + { + lights(car.x-cos(car.angle)*car.w/3-sin(car.angle)*5,car.y-sin(car.angle)*car.h/3+cos(car.angle)*5,2,C_ORANGE); + lights(car.x-cos(car.angle)*car.w/3+sin(car.angle)*5,car.y-sin(car.angle)*car.h/3-cos(car.angle)*5,2,C_ORANGE); + lights(car.x+cos(car.angle)*car.w/3-sin(car.angle)*5,car.y+sin(car.angle)*car.h/3+cos(car.angle)*5,2,C_ORANGE); + lights(car.x+cos(car.angle)*car.w/3+sin(car.angle)*5,car.y+sin(car.angle)*car.h/3-cos(car.angle)*5,2,C_ORANGE); + } + + /* display the lights */ + if (config.bynight) + { + lights(car.x+cos(car.angle)*car.w/3-sin(car.angle)*3,car.y+sin(car.angle)*car.h/3+cos(car.angle)*4,2,C_RED); + lights(car.x+cos(car.angle)*car.w/3+sin(car.angle)*3,car.y+sin(car.angle)*car.h/3-cos(car.angle)*4,2,C_RED); + lights(car.x-cos(car.angle)*car.w/3-sin(car.angle)*4,car.y-sin(car.angle)*car.h/3+cos(car.angle)*4,3,C_YELLOW); + lights(car.x-cos(car.angle)*car.w/3+sin(car.angle)*4,car.y-sin(car.angle)*car.h/3-cos(car.angle)*4,3,C_YELLOW); + pos.x=car.x-cos(car.angle)*l-sin(car.angle)*o; + pos.y=car.y-sin(car.angle)*l+cos(car.angle)*o; + pos.x-=50; + pos.y-=50; + pos.w=100; + pos.h=100; + SDL_BlitSurface(hilight,NULL,screen,&pos); + pos.x=car.x-cos(car.angle)*l+sin(car.angle)*o; + pos.y=car.y-sin(car.angle)*l-cos(car.angle)*o; + pos.x-=50; + pos.y-=50; + pos.w=100; + pos.h=100; + SDL_BlitSurface(hilight,NULL,screen,&pos); + } + + /* update display */ + if (udpsock) + { + for (i=0;iscreen->w) pos.w=screen->w-pos.x; + if (pos.y+pos.h>screen->h) pos.h=screen->h-pos.y; + SDL_UpdateRect(screen,pos.x,pos.y,pos.w,pos.h); + pos.x=car.x-cos(car.angle)*l+sin(car.angle)*15; + pos.y=car.y-sin(car.angle)*l-cos(car.angle)*15; + pos.x-=60; + pos.y-=60; + pos.w=120; + pos.h=120; + if (pos.x<0) pos.x=0; + if (pos.y<0) pos.y=0; + if (pos.x+pos.w>screen->w) pos.w=screen->w-pos.x; + if (pos.y+pos.h>screen->h) pos.h=screen->h-pos.y; + SDL_UpdateRect(screen,pos.x,pos.y,pos.w,pos.h); + } + + memcpy(oldnetpos,newnetpos,MAX_CLIENTS*sizeof(struct _car)); - /* accelerate, brake and turn depending on the pressed keys */ - if (kl) { if (speed<0) angle+=0.01*(255-b)/255*COEFF; else angle-=0.01*(255-b)/255*COEFF; } - if (kr) { if (speed<0) angle-=0.01*(255-b)/255*COEFF; else angle+=0.01*(255-b)/255*COEFF; } - if (ku) speed+=0.01*2*COEFF; - if (kd) speed-=0.01*COEFF; - /* play engine sound if no sound is currently playing */ if (lastsound_time+1000.5) || (speed>2.0 && !ku)) + if ((kd && car.speed>0.5) || (car.speed>2.0 && !ku)) { /* if the only sound is the engine, play the slide sound */ if (lastsound_time+100w/2+cos(angle)*cars[i]->w/3-sin(angle)*2,y+cars[i]->h/2+sin(angle)*cars[i]->h/3+cos(angle)*2,0); - putpixel(cir,x+cars[i]->w/2+cos(angle)*cars[i]->w/3+sin(angle)*5,y+cars[i]->h/2+sin(angle)*cars[i]->h/3-cos(angle)*5,0); + putpixel(cir,car.x+cos(car.angle)*car.w/3-sin(car.angle)*4,car.y+sin(car.angle)*car.h/3+cos(car.angle)*4,T_BLACK); + putpixel(cir,car.x+cos(car.angle)*car.w/3+sin(car.angle)*4,car.y+sin(car.angle)*car.h/3-cos(car.angle)*4,T_BLACK); /* if we are braking the slide is larger */ if (kd) { - putpixel(cir,x+cars[i]->w/2+cos(angle)*cars[i]->w/3-sin(angle)*3,y+cars[i]->h/2+sin(angle)*cars[i]->h/3+cos(angle)*3,0); - putpixel(cir,x+cars[i]->w/2+cos(angle)*cars[i]->w/3+sin(angle)*4,y+cars[i]->h/2+sin(angle)*cars[i]->h/3-cos(angle)*4,0); + putpixel(cir,car.x+cos(car.angle)*car.w/3-sin(car.angle)*3,car.y+sin(car.angle)*car.h/3+cos(car.angle)*3,T_BLACK); + putpixel(cir,car.x+cos(car.angle)*car.w/3+sin(car.angle)*3,car.y+sin(car.angle)*car.h/3-cos(car.angle)*3,T_BLACK); } } } - /* get the pixel color under the center of car in the function map */ - c=getpixel(fun,x+cars[i]->w/2,y+cars[i]->h/2); - /* red layer (checkpoints) */ - r=(c )&0xff; - /* green layer (road quality) */ - g=(c>>8 )&0xff; - /* blue layer (unused) */ - b=(c>>16)&0xff; - - /* if it is a wall we move back to the last position */ - if (g==0) - { - x=ox; - y=oy; - /* play the crash sound */ - if (lastsound_time+100MAX_LAG) + { + print(screen,WIDTH/2-strlen("Timeout !")*5,HEIGHT/2-10,"Timeout !"); + SDL_Flip(screen); + zeRace_send_time(&best); + free_mem(); + return; } - - /* look for user interaction */ - while (SDL_PollEvent(&event)) + alltime++; + + switch (car.lapflag) { - switch (event.type) - { - case SDL_QUIT: - zeRace_exit(); - break; - case SDL_KEYDOWN: - switch (event.key.keysym.sym) - { - case SDLK_ESCAPE: - /* free memory */ - Mix_FreeMusic(light); - Mix_FreeMusic(engine); - Mix_FreeMusic(crash); - Mix_FreeMusic(slide); - /* if the best time is small enought to save all keys, send it */ - if (btime<10000) zeRace_send_time(bx,by,bspeed,bangle,btime,bkeys); - return; - default: - i=event.key.keysym.sym; - if (i==config.up) ku=1; - if (i==config.down) kd=1; - if (i==config.left) kl=1; - if (i==config.right) kr=1; - break; - } - break; - case SDL_KEYUP: - i=event.key.keysym.sym; - if (i==config.up) ku=0; - if (i==config.down) kd=0; - if (i==config.left) kl=0; - if (i==config.right) kr=0; - break; - } + /* if we completed a lap */ + case 1: + printf("time = %d\"%d\n",current.time*DELAY/1000,current.time*DELAY%1000); + print(screen,0,0,"Last lap : "); + print_time(110,0,current.time); + SDL_UpdateRect(screen,0,0,200,19); + /* if it is the first turn of the best turn, save it */ + if (best.time==-1 || current.timew/2-splash->w/2-1; pos.w=splash->w+2; pos.y=screen->h/2-splash->h/2-1; pos.h=splash->h+2; - SDL_FillRect(screen,&pos,0xffffff); + SDL_FillRect(screen,&pos,C_WHITE); pos.x=screen->w/2-splash->w/2; pos.y=screen->h/2-splash->h/2; SDL_BlitSurface(splash,NULL,screen,&pos); @@ -709,7 +1057,7 @@ void zeRace_splash() /* menu loop to select track */ -void zeRace_select_track() +void zeRace_local() { SDL_Event event; @@ -717,7 +1065,7 @@ void zeRace_select_track() { SDL_Surface *full,*preview; SDL_Rect pos; - SDL_FillRect(screen,NULL,0x000000); + SDL_FillRect(screen,NULL,C_BLACK); print(screen,WIDTH/2-28*5,HEIGHT/6,"* Please choose your race *"); print(screen,WIDTH/2-strlen(tracklist->title)*5,5*HEIGHT/6-20,tracklist->title); print(screen,WIDTH/2-(strlen(tracklist->author)+strlen("Author : "))*5,5*HEIGHT/6+0,"Author : "); @@ -735,7 +1083,7 @@ void zeRace_select_track() pos.w=preview->w+2; pos.y=screen->h/2-preview->h/2-1; pos.h=preview->h+2; - SDL_FillRect(screen,&pos,0xffffff); + SDL_FillRect(screen,&pos,C_WHITE); pos.x=WIDTH/2-preview->w/2; pos.y=screen->h/2-preview->h/2; SDL_BlitSurface(preview,NULL,screen,&pos); @@ -760,7 +1108,7 @@ void zeRace_select_track() return; case SDLK_RETURN: case SDLK_SPACE: - zeRace_launch(); + zeRace_launch(0,1); update(); break; case SDLK_LEFT: @@ -782,17 +1130,291 @@ void zeRace_select_track() } +/* top 10 screen */ +void zeRace_top10(char *buf) +{ + int i,nb,tmp; + SDL_Rect pos; + SDL_FillRect(screen,NULL,C_BLACK); + nb=SDLNet_Read16(buf); + buf+=2; + print(screen,WIDTH/2-16*5,HEIGHT/14,"* Race results *"); + for (i=0;iaddress,host,port); + tmp=packet->data; + strcpy(tmp,"connect"); + tmp+=strlen(tmp)+1; + strcpy(tmp,config.pseudo); + tmp+=strlen(tmp)+1; + SDLNet_Write16(config.color,tmp); + tmp+=2; + packet->len=(void *)tmp-(void *)packet->data; + SDLNet_UDP_Send(udpsock,-1,packet); + /* network loop */ + while (SDLNet_UDP_Recv(udpsock,packet) || lagdata; + if (strcmp(tmp,"track")==0) + { + struct _tracklist *loopcheck=tracklist; + int time; + char go; + tmp+=strlen(tmp)+1; + go=*tmp++; + printf("server asked for track : %s\n",tmp); + do if (strcmp(tracklist->name,tmp)==0) break; else tracklist=tracklist->next; while (tracklist!=loopcheck); + if (strcmp(tracklist->name,tmp)!=0) + { + fprintf(stderr,"unknown track : %s\n",tmp); + zeRace_exit(); + } + tmp+=strlen(tmp)+1; + time=SDLNet_Read32(tmp); + tmp+=4; + network_speed=SDLNet_Read32(tmp); + zeRace_launch(time,go); + if (strcmp(packet->data,"finish")==0) zeRace_top10(packet->data+strlen(packet->data)+1); + lag=0; + } + SDL_Delay(7); + lag++; + } + SDLNet_UDP_Close(udpsock); + udpsock=NULL; +} + + +/* network game */ +void zeRace_network() +{ + SDL_Event event; + int active=0; + char server[MAXLINELENGTH]="localhost"; + char port[6]=PORT; + #define NETWORK_OPTIONS 4 + + void update() + { + SDL_FillRect(screen,NULL,C_BLACK); + print(screen,380,HEIGHT/(NETWORK_OPTIONS+4)*(3+active),">"); + print(screen,WIDTH/2-18*5,HEIGHT/(NETWORK_OPTIONS+4),"* Network screen *"); + print(screen,400,HEIGHT/(NETWORK_OPTIONS+4)*3,"Server : "); + print(screen,400+10*strlen("Server : "),HEIGHT/(NETWORK_OPTIONS+4)*3,server); + print(screen,400,HEIGHT/(NETWORK_OPTIONS+4)*4,"Port : "); + print(screen,400+10*strlen("Port : "),HEIGHT/(NETWORK_OPTIONS+4)*4,port); + print(screen,400,HEIGHT/(NETWORK_OPTIONS+4)*(NETWORK_OPTIONS+1),"Connect"); + print(screen,400,HEIGHT/(NETWORK_OPTIONS+4)*(NETWORK_OPTIONS+2),"Back to main menu"); + SDL_Flip(screen); + } + + update(); + for (;;) + { + while (SDL_PollEvent(&event)) + { + switch (event.type) + { + case SDL_QUIT: + zeRace_exit(); + break; + case SDL_KEYDOWN: + switch (event.key.keysym.sym) + { + case SDLK_ESCAPE: + return; + case SDLK_RETURN: + case SDLK_SPACE: + case SDLK_LEFT: + case SDLK_RIGHT: + switch (active) + { + case 0: readstring(screen,400+10*strlen("Server : "),HEIGHT/(NETWORK_OPTIONS+4)*3,server,MAXLINELENGTH); break; + case 1: readstring(screen,400+10*strlen("Port : "),HEIGHT/(NETWORK_OPTIONS+4)*4,port,5); break;; + case 2: + zeRace_connect(server,atoi(port)); + break; + case 3: + return; + } + update(); + break; + case SDLK_UP: + active--; if (active<0) active=NETWORK_OPTIONS-1; + update(); + break; + case SDLK_DOWN: + active++; if (active>NETWORK_OPTIONS-1) active=0; + update(); + break; + default: + break; + } + break; + } + } + SDL_Delay(10); + } +} + + +/* internet game */ +void zeRace_internet() +{ + IPaddress ip; + TCPsocket tcpsock; + char *request= + "GET /zerace/servers.php HTTP/1.0\n" + "Host: royale.zerezo.com\n" + "User-Agent: zeRace " VERSION "\n" + "\n"; + char response[10240],*tmp; + int len,result,i; + struct _server + { + char name[MAXLINELENGTH]; + char ip[16]; + char port[6]; + } servers[10]; + SDL_Event event; + int active=0; + #define INTERNET_OPTIONS 11 + + if (!config.internet) return; + + printf("dowloading list of servers... "); + fflush(stdout); + + if (SDLNet_ResolveHost(&ip,"royale.zerezo.com",80)==-1) + { + fprintf(stderr,"SDLNet_ResolveHost: %s\n",SDLNet_GetError()); + return; + } + + tcpsock=SDLNet_TCP_Open(&ip); + if (!tcpsock) + { + fprintf(stderr,"SDLNet_TCP_Open: %s\n",SDLNet_GetError()); + return; + } + + len=strlen(request); + result=SDLNet_TCP_Send(tcpsock,request,len); + if (result"); + print(screen,WIDTH/2-19*5,HEIGHT/(INTERNET_OPTIONS+4),"* Internet screen *"); + for (i=0;i<10;i++) + print(screen,400,HEIGHT/(INTERNET_OPTIONS+4)*(i+3),servers[i].name); + print(screen,400,HEIGHT/(INTERNET_OPTIONS+4)*(INTERNET_OPTIONS+2),"Back to main menu"); + SDL_Flip(screen); + } + + update(); + for (;;) + { + while (SDL_PollEvent(&event)) + { + switch (event.type) + { + case SDL_QUIT: + zeRace_exit(); + break; + case SDL_KEYDOWN: + switch (event.key.keysym.sym) + { + case SDLK_ESCAPE: + return; + case SDLK_RETURN: + case SDLK_SPACE: + case SDLK_LEFT: + case SDLK_RIGHT: + if (active==INTERNET_OPTIONS-1) + return; + else + zeRace_connect(servers[active].ip,atoi(servers[active].port)); + update(); + break; + case SDLK_UP: + active--; if (active<0) active=INTERNET_OPTIONS-1; + update(); + break; + case SDLK_DOWN: + active++; if (active>INTERNET_OPTIONS-1) active=0; + update(); + break; + default: + break; + } + break; + } + } + SDL_Delay(10); + } +} + + /* configuration screen */ void zeRace_config() { SDL_Event event; int active=0; - #define CONFIG_OPTIONS 11 + #define CONFIG_OPTIONS 14 void update() { SDL_Rect pos; - SDL_FillRect(screen,NULL,0x000000); + SDL_FillRect(screen,NULL,C_BLACK); print(screen,20,HEIGHT/(CONFIG_OPTIONS+4)*(3+active),">"); print(screen,WIDTH/2-24*5,HEIGHT/(CONFIG_OPTIONS+4),"* Configuration screen *"); print(screen,40,HEIGHT/(CONFIG_OPTIONS+4)*3,"Pseudo : "); @@ -816,7 +1438,13 @@ void zeRace_config() print(screen,40,HEIGHT/(CONFIG_OPTIONS+4)*12,"Color : "); pos.x=123; pos.y=HEIGHT/(CONFIG_OPTIONS+4)*12-7; - SDL_BlitSurface(cars[0],NULL,screen,&pos); + SDL_BlitSurface(cars[config.color][0],NULL,screen,&pos); + print(screen,40,HEIGHT/(CONFIG_OPTIONS+4)*13,"Boss key : "); + print(screen,40+10*strlen("Boss key : "),HEIGHT/(CONFIG_OPTIONS+4)*13,config.boss?SDL_GetKeyName(config.boss):""); + print(screen,40,HEIGHT/(CONFIG_OPTIONS+4)*14,"By night : "); + print(screen,40+10*strlen("By night : "),HEIGHT/(CONFIG_OPTIONS+4)*14,config.bynight?"Yes":"No"); + print(screen,40,HEIGHT/(CONFIG_OPTIONS+4)*15,"Internet : "); + print(screen,40+10*strlen("Internet : "),HEIGHT/(CONFIG_OPTIONS+4)*15,config.internet?"Yes":"No"); print(screen,40,HEIGHT/(CONFIG_OPTIONS+4)*(CONFIG_OPTIONS+2),"Back to main menu"); SDL_Flip(screen); } @@ -834,7 +1462,6 @@ void zeRace_config() } } - SDL_EnableKeyRepeat(SDL_DEFAULT_REPEAT_DELAY,SDL_DEFAULT_REPEAT_INTERVAL); update(); for (;;) { @@ -849,7 +1476,6 @@ void zeRace_config() switch (event.key.keysym.sym) { case SDLK_ESCAPE: - SDL_EnableKeyRepeat(0,0); return; case SDLK_RETURN: case SDLK_SPACE: @@ -868,12 +1494,13 @@ void zeRace_config() case 8: config.right=0; update(); config.right=read_key(); break; case 9: if (event.key.keysym.sym==SDLK_LEFT) config.color--; else config.color++; - if (config.color<0) config.color=11; - if (config.color>11) config.color=0; - zeRace_generate_cars(); + if (config.color<0) config.color=NB_CARS-1; + if (config.color>NB_CARS-1) config.color=0; break; - case 10: - SDL_EnableKeyRepeat(0,0); + case 10: config.boss=0; update(); config.boss=read_key(); break; + case 11: config.bynight=!config.bynight; break; + case 12: config.internet=!config.internet; break; + case 13: return; } update(); @@ -903,26 +1530,26 @@ void zeRace_menu() SDL_Event event; int active=0; SDL_Surface *logo; - #define MENU_OPTIONS 3 + #define MENU_OPTIONS 5 void update() { SDL_Rect pos; - SDL_FillRect(screen,NULL,0x000000); + SDL_FillRect(screen,NULL,C_BLACK); pos.x=WIDTH/2-logo->w/2; pos.y=HEIGHT/6-logo->h/2; SDL_BlitSurface(logo,NULL,screen,&pos); - print(screen,650,HEIGHT/(MENU_OPTIONS+4)*2,"version " VERSION); + print(screen,650,HEIGHT/6+logo->h/3,"version " VERSION); print(screen,420,HEIGHT/(MENU_OPTIONS+4)*(3+active),">"); print(screen,440,HEIGHT/(MENU_OPTIONS+4)*3,"Local game"); - /* print(screen,440,HEIGHT/(MENU_OPTIONS+4)*4,"Network game"); */ - print(screen,440,HEIGHT/(MENU_OPTIONS+4)*4,"Configuration"); - print(screen,440,HEIGHT/(MENU_OPTIONS+4)*5,"Exit game"); + print(screen,440,HEIGHT/(MENU_OPTIONS+4)*4,"Network game"); + print(screen,440,HEIGHT/(MENU_OPTIONS+4)*5,"Internet game"); + print(screen,440,HEIGHT/(MENU_OPTIONS+4)*6,"Configuration"); + print(screen,440,HEIGHT/(MENU_OPTIONS+4)*7,"Exit game"); SDL_Flip(screen); } logo=IMG_Load("sprites/logo.jpg"); - SDL_EnableKeyRepeat(SDL_DEFAULT_REPEAT_DELAY,SDL_DEFAULT_REPEAT_INTERVAL); update(); for (;;) { @@ -937,7 +1564,6 @@ void zeRace_menu() switch (event.key.keysym.sym) { case SDLK_ESCAPE: - SDL_EnableKeyRepeat(0,0); return; case SDLK_RETURN: case SDLK_SPACE: @@ -945,9 +1571,11 @@ void zeRace_menu() case SDLK_RIGHT: switch (active) { - case 0: zeRace_select_track(); break; - case 1: zeRace_config(); break; - case 2: return; + case 0: zeRace_local(); break; + case 1: zeRace_network(); break; + case 2: zeRace_internet(); break; + case 3: zeRace_config(); break; + case 4: return; } update(); break;