SDL 1.2 Придется дважды нажать клавишу, чтобы события вступили в силу
У меня есть петля в моей главной, чтобы ловить события. Если игрок набирает "1", то я вызываю функцию play(), которая также содержит цикл для событий (для ходов и т. Д.). Я действительно не понимаю почему, но мне нужно дважды нажать ESCAPE, чтобы выйти из игры, а другие просто не распознаются. Если я удаляю свои петли в свою главную, тогда все идет хорошо.
Вот мой цикл в основном:
while(continuer)
{
SDL_WaitEvent(&event);
switch(event.type)
{
case SDL_QUIT:
continuer = 0;
break;
case SDL_KEYDOWN:
switch(event.key.keysym.sym)
{
case SDLK_ESCAPE:
continuer = 0;
break;
case SDLK_KP1:
play(screen);
break;
}
break;
}
}
А вот и моя функция (она все еще в процессе, так что не шокируйся, если она покажется ОЧЕНЬ глупой:D)
void play(SDL_Surface* screen)
{
int i=0, j=0, continuer=1;
int elements[LARGEUR_MAP][HAUTEUR_MAP];
char object =' ';
FILE* level = NULL;
SDL_Surface *lvl[LARGEUR_MAP][HAUTEUR_MAP],*mario[4];
SDL_Rect posElem, posMario;
SDL_Event event;
screen = SDL_SetVideoMode(LARGEUR_FENETRE, HAUTEUR_FENETRE,32,SDL_HWSURFACE | SDL_DOUBLEBUF);
SDL_FillRect(screen, NULL, SDL_MapRGB(screen->format,255,255,255));
mario[HAUT] = IMG_Load("images\\mario_haut.gif");
mario[DROITE] = IMG_Load("images\\mario_droite.gif");
mario[BAS] = IMG_Load("images\\mario_bas.gif");
mario[GAUCHE] = IMG_Load("images\\mario_gauche.gif");
//Ouverture du fichier contenant les infos du niveau
level = fopen("lvl.txt","r");
if(level == NULL)
{
fprintf(stderr,"Erreur lors de l'ouverture du fichier");
exit(EXIT_FAILURE);
}
//Boucle pour lire le fichier et placer les éléments du décor
for(j=0;j<HAUTEUR_MAP;j++)
{
for(i=0;i<LARGEUR_MAP;i++)
{
object = fgetc(level);
if(object == '\n')
object = fgetc(level);
switch(object)
{
case 'm':
lvl[i][j] = IMG_Load("images\\mur.jpg");
elements[i][j] = MUR;
posElem.x = i*TAILLE_BLOC;
posElem.y = j*TAILLE_BLOC;
SDL_BlitSurface(lvl[i][j], NULL, screen, &posElem);
break;
case 'o':
lvl[i][j] = IMG_Load("images\\objectif.png");
elements[i][j] = OBJECTIF;
posElem.x = i*TAILLE_BLOC;
posElem.y = j*TAILLE_BLOC;
SDL_BlitSurface(lvl[i][j], NULL, screen, &posElem);
break;
case 'c':
lvl[i][j] = IMG_Load("images\\caisse.jpg");
elements[i][j] = CAISSE;
posElem.x = i*TAILLE_BLOC;
posElem.y = j*TAILLE_BLOC;
SDL_BlitSurface(lvl[i][j], NULL, screen, &posElem);
break;
case 'M':
lvl[i][j] = IMG_Load("images\\mario_bas.gif");
elements[i][j] = MARIO;
posElem.x = i*TAILLE_BLOC;
posElem.y = j*TAILLE_BLOC;
SDL_BlitSurface(lvl[i][j], NULL, screen, &posElem);
break;
default :
elements[i][j] = VIDE;
break;
}
}
}
SDL_Flip(screen);
fclose(level);
//Code à l'arrache... Je met en dur la position de Mario, on verra plus tard pour l'automatisation ~~
i = 5;
j = 2;
posMario.x = i * TAILLE_BLOC;
posMario.y = j * TAILLE_BLOC;
while(continuer)
{
SDL_WaitEvent(&event);
switch(event.type)
{
case SDL_QUIT:
continuer = 0;
break;
case SDL_KEYDOWN:
switch(event.key.keysym.sym)
{
case SDLK_UP:
continuer = 0;
if(elements[i-1][j] != MUR && elements[i-1][j] != CAISSE_OK && i != 0)
{
posMario.x = (i-1)*TAILLE_BLOC;
posMario.y = j*TAILLE_BLOC;
if(elements[i-1][j] == CAISSE && elements[i-2][j] != MUR && elements[i-2][j] != CAISSE && elements[i-2][j] != CAISSE_OK && (i-2)>0)
{
SDL_FillRect(lvl[i][j], NULL, SDL_MapRGB(screen->format,255,255,255));
SDL_BlitSurface(mario[HAUT],NULL,screen,&posMario);
SDL_Flip(screen);
}
}
break;
case SDLK_ESCAPE:
continuer = 0;
break;
}
}
}
}
1 ответ
Насколько я понял, у вас есть два (вложенных) цикла событий, каждый из которых использует свою локальную переменную продолжателя. Когда вы вошли во внутренний цикл, нажатие escape выходит из этого внешнего: вам нужно снова нажать esc, чтобы выйти из этого цикла.
В зависимости от того, что вы хотите, одним из возможных быстрых решений может быть, например, возврат true или false из play(), чтобы указать, хочет ли игрок выйти из игры. Лично я думаю, что правильное поведение вложенных циклов событий может быть проблематичным (когда они большие и глубокие), но, к сожалению, во многих случаях вы не можете избежать их. Будь осторожен с ними, если сможешь.