mirror of
https://gitlab.com/dosowisko.net/libsuperderpy.git
synced 2025-01-10 08:47:59 +01:00
396 lines
15 KiB
C
396 lines
15 KiB
C
/*! \file level.c
|
|
* \brief Playable Level code.
|
|
*/
|
|
/*
|
|
* Copyright (c) Sebastian Krzyszkowiak <dos@dosowisko.net>
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
*/
|
|
#include <stdio.h>
|
|
#include "moonwalk.h"
|
|
#include "config.h"
|
|
#include "pause.h"
|
|
#include "level.h"
|
|
#include "timeline.h"
|
|
|
|
// TODO: fix spritesheet speed to not depend on FPS
|
|
|
|
void Level_Passed(struct Game *game) {
|
|
if (game->level.current_level<6) {
|
|
int available = atoi(GetConfigOptionDefault("MuffinAttack", "level", "1"));
|
|
available++;
|
|
if ((available<2) || (available>7)) available=1;
|
|
if (available==(game->level.current_level+1)) {
|
|
char* text = malloc(2*sizeof(char));
|
|
sprintf(text, "%d", available);
|
|
SetConfigOption("MuffinAttack", "level", text);
|
|
free(text);
|
|
}
|
|
} else {
|
|
SetConfigOption("MuffinAttack", "completed", "1");
|
|
}
|
|
}
|
|
|
|
bool Accelerate(struct Game *game, struct TM_Action *action, enum TM_ActionState state) {
|
|
if (state != TM_ACTIONSTATE_RUNNING) return false;
|
|
game->level.speed+=tps(game, 0.000005*60);
|
|
if (game->level.speed<tps(game, 0.0020*60)) return false;
|
|
return true;
|
|
}
|
|
|
|
bool Walk(struct Game *game, struct TM_Action *action, enum TM_ActionState state) {
|
|
if (state != TM_ACTIONSTATE_RUNNING) return false;
|
|
game->level.derpy_x+=tps(game, 60*0.001);
|
|
if (game->level.derpy_x<0.05) return false;
|
|
return true;
|
|
}
|
|
|
|
bool Move(struct Game *game, struct TM_Action *action, enum TM_ActionState state) {
|
|
if (state != TM_ACTIONSTATE_RUNNING) return false;
|
|
game->level.speed=tps(game, 60*0.00035);
|
|
if (game->level.st_pos<0.275) return false;
|
|
return true;
|
|
}
|
|
|
|
bool Fly(struct Game *game, struct TM_Action *action, enum TM_ActionState state) {
|
|
if (state != TM_ACTIONSTATE_RUNNING) return false;
|
|
game->level.derpy_y-=tps(game, 60*0.004);
|
|
if (game->level.derpy_y>0.2) return false;
|
|
game->level.handle_input=true;
|
|
return true;
|
|
}
|
|
|
|
|
|
bool Stop(struct Game *game, struct TM_Action *action, enum TM_ActionState state) {
|
|
if (state != TM_ACTIONSTATE_RUNNING) return false;
|
|
game->level.speed=0;
|
|
game->level.sheet_speed = 0;
|
|
//game->level.sheet_pos = 0;
|
|
return true;
|
|
}
|
|
|
|
bool Letter(struct Game *game, struct TM_Action *action, enum TM_ActionState state) {
|
|
if (state != TM_ACTIONSTATE_RUNNING) return false;
|
|
al_draw_text_with_shadow(game->menu.font_title, al_map_rgb(255,255,255), al_get_display_width(game->display)*0.5, al_get_display_height(game->display)*0.45, ALLEGRO_ALIGN_CENTRE, "Letter from Twilight");
|
|
struct ALLEGRO_KEYBOARD_STATE keyboard;
|
|
al_get_keyboard_state(&keyboard);
|
|
if (al_key_down(&keyboard, ALLEGRO_KEY_ENTER)) {
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void Level_Draw(struct Game *game) {
|
|
if (!al_get_sample_instance_playing(game->level.music) && (game->loadstate==GAMESTATE_LEVEL)) {
|
|
al_set_sample_instance_playing(game->level.music, true);
|
|
al_set_sample_instance_position(game->level.music, game->level.music_pos);
|
|
}
|
|
if (game->level.current_level!=1) Moonwalk_Draw(game);
|
|
else {
|
|
|
|
struct ALLEGRO_KEYBOARD_STATE keyboard;
|
|
al_get_keyboard_state(&keyboard);
|
|
if (game->level.handle_input) {
|
|
if (al_key_down(&keyboard, ALLEGRO_KEY_UP)) {
|
|
game->level.derpy_y -= 0.005;
|
|
//PrintConsole(game, "Derpy Y position: %f", game->level.derpy_y);
|
|
}
|
|
if (al_key_down(&keyboard, ALLEGRO_KEY_DOWN)) {
|
|
game->level.derpy_y += 0.005;
|
|
//PrintConsole(game, "Derpy Y position: %f", game->level.derpy_y);
|
|
}
|
|
if (game->level.derpy_y > 0.5) game->level.sheet_speed = 0.0020/game->level.speed;
|
|
else game->level.sheet_speed=0;
|
|
if (game->level.derpy_y < 0) game->level.derpy_y=0;
|
|
else if (game->level.derpy_y > 0.75) game->level.derpy_y=0.75;
|
|
}
|
|
|
|
al_clear_to_color(al_map_rgb(0,0,0));
|
|
al_draw_bitmap(game->level.clouds, (-game->level.cl_pos)*al_get_bitmap_width(game->level.clouds), 0, 0);
|
|
al_draw_bitmap(game->level.clouds, (1+(-game->level.cl_pos))*al_get_bitmap_width(game->level.clouds), 0, 0);
|
|
al_draw_bitmap(game->level.background, (-game->level.bg_pos)*al_get_bitmap_width(game->level.background), 0, 0);
|
|
al_draw_bitmap(game->level.background, (1+(-game->level.bg_pos))*al_get_bitmap_width(game->level.background), 0, 0);
|
|
al_draw_bitmap(game->level.stage, (-game->level.st_pos)*al_get_bitmap_width(game->level.stage), 0 ,0);
|
|
al_draw_bitmap(game->level.stage, (1+(-game->level.st_pos))*al_get_bitmap_width(game->level.stage), 0 ,0);
|
|
|
|
al_set_target_bitmap(game->level.derpy);
|
|
al_clear_to_color(al_map_rgba(0,0,0,0));
|
|
al_draw_bitmap_region(game->level.derpy_walkcycle,al_get_bitmap_width(game->level.derpy)*(game->level.sheet_pos%game->level.sheet_cols),al_get_bitmap_height(game->level.derpy)*(game->level.sheet_pos/game->level.sheet_cols),al_get_bitmap_width(game->level.derpy), al_get_bitmap_height(game->level.derpy),0,0,0);
|
|
if (game->level.sheet_speed) {
|
|
//int i;
|
|
//for (i = 0; i < tps(game, 60); i++ ) {
|
|
game->level.sheet_tmp+=tps(game, 60);
|
|
//if (game->level.sheet_speed==1) game->level.sheet_pos++;
|
|
if (game->level.sheet_tmp >= game->level.sheet_speed) {
|
|
game->level.sheet_pos++;
|
|
game->level.sheet_tmp = 0;
|
|
//game->level.sheet_tmp -= game->level.sheet_speed;
|
|
}
|
|
if (game->level.sheet_pos>=game->level.sheet_cols*game->level.sheet_rows) game->level.sheet_pos=0;
|
|
//}
|
|
}
|
|
al_set_target_bitmap(al_get_backbuffer(game->display));
|
|
|
|
al_draw_bitmap(game->level.derpy, game->level.derpy_x*al_get_display_width(game->display), game->level.derpy_y*al_get_display_height(game->display), 0);
|
|
|
|
al_draw_bitmap(game->level.foreground, (-game->level.fg_pos)*al_get_bitmap_width(game->level.foreground), 0 ,0);
|
|
al_draw_bitmap(game->level.foreground, (1+(-game->level.fg_pos))*al_get_bitmap_width(game->level.foreground), 0 ,0);
|
|
if (game->level.speed > 0) {
|
|
game->level.cl_pos += game->level.speed * 0.2;
|
|
game->level.bg_pos += game->level.speed * 0.6;
|
|
game->level.st_pos += game->level.speed * 1;
|
|
game->level.fg_pos += game->level.speed * 1.75;
|
|
if (game->level.bg_pos >= 1) game->level.bg_pos=game->level.bg_pos-1;
|
|
if (game->level.st_pos >= 1) game->level.st_pos=game->level.st_pos-1;
|
|
if (game->level.fg_pos >= 1) game->level.fg_pos=game->level.fg_pos-1;
|
|
}
|
|
game->level.cl_pos += tps(game, 60*0.00005);
|
|
if (game->level.cl_pos >= 1) game->level.cl_pos=game->level.cl_pos-1;
|
|
TM_Process();
|
|
}
|
|
}
|
|
|
|
bool FadeIn(struct Game *game, struct TM_Action *action, enum TM_ActionState state) {
|
|
if (!action->arguments) {
|
|
action->arguments = TM_AddToArgs(action->arguments, malloc(sizeof(float)));
|
|
action->arguments = TM_AddToArgs(action->arguments, (void*)al_create_bitmap(al_get_display_width(game->display), al_get_display_height(game->display)));
|
|
}
|
|
float* fadeloop;
|
|
ALLEGRO_BITMAP* fade_bitmap;
|
|
fadeloop = (float*)action->arguments->value;
|
|
fade_bitmap = (ALLEGRO_BITMAP*)action->arguments->next->value;
|
|
if (state == TM_ACTIONSTATE_INIT) {
|
|
*fadeloop = 255;
|
|
al_set_target_bitmap(fade_bitmap);
|
|
al_clear_to_color(al_map_rgb(0,0,0));
|
|
al_set_target_bitmap(al_get_backbuffer(game->display));
|
|
} else if (state == TM_ACTIONSTATE_RUNNING) {
|
|
al_draw_tinted_bitmap(fade_bitmap,al_map_rgba_f(1,1,1,*fadeloop/255.0),0,0,0);
|
|
*fadeloop-=tps(game, 600);
|
|
if (*fadeloop<=0) return true;
|
|
} else {
|
|
al_destroy_bitmap(fade_bitmap);
|
|
free(fadeloop);
|
|
TM_DestroyArgs(action->arguments);
|
|
action->arguments = NULL;
|
|
al_play_sample_instance(game->level.music);
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool FadeOut(struct Game *game, struct TM_Action *action, enum TM_ActionState state) {
|
|
if (!action->arguments) {
|
|
action->arguments = TM_AddToArgs(action->arguments, malloc(sizeof(float)));
|
|
action->arguments = TM_AddToArgs(action->arguments, (void*)al_create_bitmap(al_get_display_width(game->display), al_get_display_height(game->display)));
|
|
}
|
|
float* fadeloop;
|
|
ALLEGRO_BITMAP* fade_bitmap;
|
|
fadeloop = (float*)action->arguments->value;
|
|
fade_bitmap = (ALLEGRO_BITMAP*)action->arguments->next->value;
|
|
if (state == TM_ACTIONSTATE_INIT) {
|
|
*fadeloop = 0;
|
|
al_set_target_bitmap(fade_bitmap);
|
|
al_clear_to_color(al_map_rgb(0,0,0));
|
|
al_set_target_bitmap(al_get_backbuffer(game->display));
|
|
} else if (state == TM_ACTIONSTATE_RUNNING) {
|
|
al_draw_tinted_bitmap(fade_bitmap,al_map_rgba_f(1,1,1,*fadeloop/255.0),0,0,0);
|
|
*fadeloop+=tps(game, 600);
|
|
if (*fadeloop>=256) return true;
|
|
} else {
|
|
al_destroy_bitmap(fade_bitmap);
|
|
free(fadeloop);
|
|
Level_Unload(game);
|
|
game->gamestate = GAMESTATE_LOADING;
|
|
game->loadstate = GAMESTATE_MAP;
|
|
TM_DestroyArgs(action->arguments);
|
|
action->arguments = NULL;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool Welcome(struct Game *game, struct TM_Action *action, enum TM_ActionState state) {
|
|
float* tmp; bool* in;
|
|
if (!action->arguments) {
|
|
action->arguments = TM_AddToArgs(action->arguments, malloc(sizeof(float)));
|
|
action->arguments = TM_AddToArgs(action->arguments, malloc(sizeof(bool)));
|
|
}
|
|
tmp = (float*)action->arguments->value;
|
|
in = (bool*)action->arguments->next->value;
|
|
if (state == TM_ACTIONSTATE_INIT) {
|
|
*tmp = 0;
|
|
*in = true;
|
|
//PrintConsole(game, "WELCOME INIT");
|
|
}
|
|
else if (state == TM_ACTIONSTATE_RUNNING) {
|
|
//al_draw_text_with_shadow(game->font, al_map_rgb(255,255,255), *tmp, 20, 0, "Level 1");
|
|
//PrintConsole(game, "WELCOME RUNNING FADE=%f, IN=%d", *in);
|
|
int fade = *tmp;
|
|
if (fade>255) fade=255;
|
|
if (*tmp > 2048) { *tmp=255; *in=false; }
|
|
al_draw_tinted_bitmap(game->level.welcome, al_map_rgba_f(fade/255.0,fade/255.0,fade/255.0,fade/255.0), 0, 0, 0);
|
|
if (*in) {
|
|
*tmp+=tps(game, 600);
|
|
//if (*tmp>=256) { return false; }
|
|
} else {
|
|
*tmp-=tps(game, 600);
|
|
if (*tmp<=0) { return true; }
|
|
}
|
|
} else {
|
|
free(action->arguments->value);
|
|
free(action->arguments->next->value);
|
|
TM_DestroyArgs(action->arguments);
|
|
action->arguments = NULL;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool PassLevel(struct Game *game, struct TM_Action *action, enum TM_ActionState state) {
|
|
if (state == TM_ACTIONSTATE_DESTROY) {
|
|
Level_Passed(game);
|
|
TM_AddBackgroundAction(&FadeOut, NULL, 0, "fadeout");
|
|
}
|
|
return true;
|
|
}
|
|
|
|
void Level_Load(struct Game *game) {
|
|
game->level.cl_pos=0;
|
|
game->level.bg_pos=0;
|
|
game->level.fg_pos=0.2;
|
|
game->level.st_pos=0.1;
|
|
game->level.speed = 0;
|
|
game->level.derpy_x = -0.2;
|
|
game->level.derpy_y = 0.6;
|
|
game->level.sheet_rows = 4;
|
|
game->level.sheet_cols = 6;
|
|
game->level.sheet_pos = 0;
|
|
game->level.sheet_speed = 2.4;
|
|
game->level.sheet_tmp = 0;
|
|
game->level.handle_input = false;
|
|
al_clear_to_color(al_map_rgb(0,0,0));
|
|
if (game->level.current_level!=1) Moonwalk_Load(game);
|
|
else {
|
|
TM_Init(game);
|
|
TM_AddBackgroundAction(&FadeIn, NULL, 0, "fadein");
|
|
TM_AddDelay(1000);
|
|
TM_AddQueuedBackgroundAction(&Welcome, NULL, 0, "welcome");
|
|
TM_AddDelay(1000);
|
|
TM_AddAction(&Walk, NULL, "walk");
|
|
TM_AddAction(&Move, NULL, "move");
|
|
TM_AddAction(&Stop, NULL, "stop");
|
|
TM_AddDelay(1000);
|
|
TM_AddAction(&Letter, NULL, "letter");
|
|
TM_AddDelay(500);
|
|
TM_AddQueuedBackgroundAction(&Accelerate, NULL, 0, "accelerate");
|
|
TM_AddAction(&Fly, NULL, "fly");
|
|
// Derpy walks in... (background - owl)
|
|
// Derpy reads a letter
|
|
// Letter on screen
|
|
// Derpy: fly! (background - accelerate)
|
|
|
|
// first part gameplay goes here
|
|
TM_AddDelay(60*1000);
|
|
|
|
// wings disappear, deccelerate, fall down
|
|
// run
|
|
// show Fluttershy's house
|
|
|
|
// second part gameplay goes here
|
|
//
|
|
|
|
// cutscene goes here
|
|
//
|
|
|
|
TM_AddAction(&PassLevel, NULL, "passlevel");
|
|
}
|
|
}
|
|
|
|
int Level_Keydown(struct Game *game, ALLEGRO_EVENT *ev) {
|
|
if (ev->keyboard.keycode==ALLEGRO_KEY_ESCAPE) {
|
|
game->level.music_pos = al_get_sample_instance_position(game->level.music);
|
|
al_set_sample_instance_playing(game->level.music, false);
|
|
}
|
|
if (game->level.current_level!=1) Moonwalk_Keydown(game, ev);
|
|
if (ev->keyboard.keycode==ALLEGRO_KEY_ESCAPE) {
|
|
game->gamestate = GAMESTATE_PAUSE;
|
|
game->loadstate = GAMESTATE_LEVEL;
|
|
Pause_Load(game);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
void Level_ProcessLogic(struct Game *game, ALLEGRO_EVENT *ev) {
|
|
if (game->level.current_level==1) TM_HandleEvent(ev);
|
|
|
|
}
|
|
|
|
void Level_Preload(struct Game *game) {
|
|
Pause_Preload(game);
|
|
if (game->level.current_level!=1) Moonwalk_Preload(game);
|
|
else {
|
|
game->level.sample = al_load_sample( "data/levels/1/music.flac" );
|
|
game->level.music = al_create_sample_instance(game->level.sample);
|
|
al_attach_sample_instance_to_mixer(game->level.music, game->audio.music);
|
|
al_set_sample_instance_playmode(game->level.music, ALLEGRO_PLAYMODE_LOOP);
|
|
|
|
if (!game->level.sample){
|
|
fprintf(stderr, "Audio clip sample not loaded!\n" );
|
|
exit(-1);
|
|
}
|
|
}
|
|
Level_PreloadBitmaps(game);
|
|
}
|
|
|
|
void Level_Unload(struct Game *game) {
|
|
Pause_Unload_Real(game);
|
|
al_destroy_sample_instance(game->level.music);
|
|
al_destroy_sample(game->level.sample);
|
|
if (game->level.current_level!=1) Moonwalk_Unload(game);
|
|
else {
|
|
TM_Destroy();
|
|
}
|
|
Level_UnloadBitmaps(game);
|
|
}
|
|
|
|
void Level_UnloadBitmaps(struct Game *game) {
|
|
al_destroy_bitmap(game->level.derpy);
|
|
al_destroy_bitmap(game->level.derpy_walkcycle);
|
|
if (game->level.current_level!=1) Moonwalk_UnloadBitmaps(game);
|
|
else {
|
|
al_destroy_bitmap(game->level.foreground);
|
|
al_destroy_bitmap(game->level.background);
|
|
al_destroy_bitmap(game->level.stage);
|
|
al_destroy_bitmap(game->level.clouds);
|
|
al_destroy_bitmap(game->level.welcome);
|
|
}
|
|
}
|
|
|
|
void Level_PreloadBitmaps(struct Game *game) {
|
|
game->level.derpy_walkcycle = LoadScaledBitmap("derpcycle.png", al_get_display_width(game->display)*0.1953125*6, al_get_display_height(game->display)*0.25*4);
|
|
game->level.derpy = al_create_bitmap(al_get_display_width(game->display)*0.1953125, al_get_display_height(game->display)*0.25);
|
|
if (game->level.current_level!=1) Moonwalk_PreloadBitmaps(game);
|
|
else {
|
|
// TODO: handle strange display aspects
|
|
game->level.clouds = LoadScaledBitmap("levels/1/clouds.png", al_get_display_height(game->display)*4.73307291666666666667, al_get_display_height(game->display));
|
|
game->level.foreground = LoadScaledBitmap("levels/1/foreground.png", al_get_display_height(game->display)*4.73307291666666666667, al_get_display_height(game->display));
|
|
game->level.background = LoadScaledBitmap("levels/1/background.png", al_get_display_height(game->display)*4.73307291666666666667, al_get_display_height(game->display));
|
|
game->level.stage = LoadScaledBitmap("levels/1/stage.png", al_get_display_height(game->display)*4.73307291666666666667, al_get_display_height(game->display));
|
|
game->level.welcome = al_create_bitmap(al_get_display_width(game->display), al_get_display_height(game->display));
|
|
al_set_target_bitmap(game->level.welcome);
|
|
al_clear_to_color(al_map_rgba(0,0,0,0));
|
|
al_draw_text_with_shadow(game->menu.font_title, al_map_rgb(255,255,255), al_get_display_width(game->display)*0.5, al_get_display_height(game->display)*0.1, ALLEGRO_ALIGN_CENTRE, "Level 1");
|
|
al_draw_text_with_shadow(game->menu.font_subtitle, al_map_rgb(255,255,255), al_get_display_width(game->display)*0.5, al_get_display_height(game->display)*0.275, ALLEGRO_ALIGN_CENTRE, "Fluttershy");
|
|
al_set_target_bitmap(al_get_backbuffer(game->display));
|
|
}
|
|
}
|