Add level preview generation

main
En Yi 2025-07-30 21:57:13 +08:00
parent e90856f8c6
commit c2fe0fda8a
2 changed files with 142 additions and 6 deletions

View File

@ -4,27 +4,156 @@
#include "raymath.h" #include "raymath.h"
#include <stdio.h> #include <stdio.h>
#define LEVEL_PREVIEW_SIZE 400
static void level_select_render_func(Scene_t* scene) static void level_select_render_func(Scene_t* scene)
{ {
LevelSelectSceneData_t* data = &(CONTAINER_OF(scene, LevelSelectScene_t, scene)->data); LevelSelectSceneData_t* data = &(CONTAINER_OF(scene, LevelSelectScene_t, scene)->data);
Sprite_t* level_board = get_sprite(&scene->engine->assets, "lvl_board"); Sprite_t* level_board = get_sprite(&scene->engine->assets, "lvl_board");
Sprite_t* level_select = get_sprite(&scene->engine->assets, "lvl_select"); Sprite_t* level_select = get_sprite(&scene->engine->assets, "lvl_select");
Sprite_t* preview = get_sprite(&scene->engine->assets, "lvlprvw"); //Sprite_t* preview = get_sprite(&scene->engine->assets, "lvlprvw");
Font* menu_font = get_font(&scene->engine->assets, "MenuFont"); Font* menu_font = get_font(&scene->engine->assets, "MenuFont");
BeginTextureMode(scene->layers.render_layers[0].layer_tex); BeginTextureMode(scene->layers.render_layers[0].layer_tex);
ClearBackground(BLANK); ClearBackground(BLANK);
draw_sprite(level_select, 0, (Vector2){0,0},0, false); draw_sprite(level_select, 0, (Vector2){0,0},0, false);
draw_sprite(level_board, 0, (Vector2){level_select->frame_size.x,0},0, false); draw_sprite(level_board, 0, (Vector2){level_select->frame_size.x,0},0, false);
draw_sprite(preview, data->scroll_area.curr_selection, (Vector2){ Rectangle draw_rec = {0,0,LEVEL_PREVIEW_SIZE,LEVEL_PREVIEW_SIZE * -1};
level_select->frame_size.x + (level_board->frame_size.x - preview->frame_size.x) / 2, Vector2 draw_pos = {
(level_board->frame_size.y - preview->frame_size.y) / 2, level_select->frame_size.x + (level_board->frame_size.x - LEVEL_PREVIEW_SIZE) / 2,
},0, false); (level_board->frame_size.y - LEVEL_PREVIEW_SIZE) / 2
};
DrawTextureRec(
data->preview.texture,
draw_rec,
draw_pos,
WHITE
);
DrawTextEx(*menu_font, "Level Select", (Vector2){60, 20}, 40, 4, BLACK); DrawTextEx(*menu_font, "Level Select", (Vector2){60, 20}, 40, 4, BLACK);
vert_scrollarea_render(&data->scroll_area); vert_scrollarea_render(&data->scroll_area);
EndTextureMode(); EndTextureMode();
} }
static void level_preview_render_func(Scene_t* scene)
{
LevelSelectSceneData_t* data = &(CONTAINER_OF(scene, LevelSelectScene_t, scene)->data);
if (!data->update_preview) return;
LevelMap_t level = data->level_pack->levels[data->scroll_area.curr_selection];
const uint32_t n_tiles = level.width * level.height;
uint16_t max_dim = (level.width > level.height ? level.width : level.height);
max_dim = (max_dim == 0)? 1: max_dim;
uint32_t tile_size = LEVEL_PREVIEW_SIZE / max_dim;
uint32_t tile_halfsize = tile_size >> 1;
tile_halfsize = (tile_halfsize == 0)? 1 : tile_halfsize;
const uint32_t lvl_width = level.width*tile_size-1;
const uint32_t lvl_height = level.height*tile_size-1;
const uint32_t x_offset = (LEVEL_PREVIEW_SIZE - lvl_width) / 2;
const uint32_t y_offset = (LEVEL_PREVIEW_SIZE - lvl_height) / 2;
const Color danger_col = {239,79,81,255};
BeginTextureMode(data->preview);
ClearBackground((Color){255,255,255,0});
DrawRectangle(
x_offset, y_offset, lvl_width, lvl_height,
(Color){64,64,64,255}
);
for (uint32_t i = 0; i < n_tiles; ++i)
{
uint32_t pos_x = tile_size * (i % level.width) + x_offset;
uint32_t pos_y = tile_size * (i / level.width) + y_offset;
if (level.tiles[i].tile_type >= 8 && level.tiles[i].tile_type < 20)
{
uint32_t tmp_idx = level.tiles[i].tile_type - 8;
uint32_t item_type = tmp_idx % 6;
Color col = (tmp_idx > 5)? (Color){110,110,110,255} : (Color){160,117,48,255};
DrawRectangle(pos_x, pos_y, tile_size, tile_size, col);
switch (item_type)
{
case 1:
DrawLine(pos_x, pos_y + tile_halfsize, pos_x + tile_halfsize, pos_y + tile_halfsize, danger_col);
break;
case 2:
DrawLine(pos_x + tile_halfsize, pos_y + tile_halfsize, pos_x + tile_size, pos_y + tile_halfsize, danger_col);
break;
case 3:
DrawLine(pos_x + tile_halfsize, pos_y, pos_x + tile_halfsize, pos_y + tile_halfsize, danger_col);
break;
case 4:
DrawLine(pos_x + tile_halfsize, pos_y + tile_halfsize, pos_x + tile_halfsize, pos_y + tile_size, danger_col);
break;
case 5:
DrawLine(pos_x, pos_y, pos_x + tile_size, pos_y + tile_size, danger_col);
DrawLine(pos_x, pos_y + tile_size, pos_x + tile_size, pos_y, danger_col);
break;
}
DrawRectangleLines(pos_x, pos_y, tile_size, tile_size, (Color){0,0,0,64});
}
else
{
switch(level.tiles[i].tile_type) {
case SOLID_TILE:
DrawRectangle(pos_x, pos_y, tile_size, tile_size, BLACK);
break;
case ONEWAY_TILE:
DrawRectangle(pos_x, pos_y, tile_size, tile_halfsize, (Color){128,64,0,255});
break;
case LADDER:
DrawRectangleLines(pos_x, pos_y, tile_size, tile_size, (Color){214,141,64,255});
break;
case 4:
case 5:
case 6:
case 7:
// Copied from level generation
// Priority: Down, Up, Left, Right
if (i + level.width < n_tiles && level.tiles[i + level.width].tile_type == SOLID_TILE)
{
DrawRectangle(pos_x, pos_y + tile_halfsize , tile_size, tile_halfsize, danger_col);
}
else if (i >= level.width && level.tiles[i - level.width].tile_type == SOLID_TILE)
{
DrawRectangle(pos_x, pos_y, tile_size, tile_halfsize, danger_col);
}
else if (i % level.width != 0 && level.tiles[i - 1].tile_type == SOLID_TILE)
{
DrawRectangle(pos_x, pos_y, tile_halfsize, tile_size, danger_col);
}
else if ((i + 1) % level.width != 0 && level.tiles[i + 1].tile_type == SOLID_TILE)
{
DrawRectangle(pos_x + tile_halfsize, pos_y, tile_halfsize, tile_size, danger_col);
}
else
{
DrawRectangle(pos_x, pos_y + tile_halfsize , tile_size, tile_halfsize, danger_col);
}
break;
case 20:
DrawCircle(pos_x + tile_halfsize, pos_y + tile_halfsize, tile_halfsize, (Color){12,12,12,255});
break;
case 22:
DrawRectangle(pos_x, pos_y, tile_size, tile_size, (Color){255,0,255,255});
break;
case 23:
DrawRectangle(pos_x, pos_y, tile_size, tile_size, (Color){255,255,0,255});
break;
case 24:
DrawRectangle(pos_x, pos_y, tile_size, tile_size, (Color){0,255,0,255});
break;
case 25:
DrawCircle(pos_x + tile_halfsize, pos_y + tile_halfsize, tile_halfsize-1, danger_col);
break;
}
}
}
EndTextureMode();
data->update_preview = false;
}
static void level_select_do_action(Scene_t* scene, ActionType_t action, bool pressed) static void level_select_do_action(Scene_t* scene, ActionType_t action, bool pressed)
{ {
LevelSelectSceneData_t* data = &(CONTAINER_OF(scene, LevelSelectScene_t, scene)->data); LevelSelectSceneData_t* data = &(CONTAINER_OF(scene, LevelSelectScene_t, scene)->data);
@ -37,6 +166,7 @@ static void level_select_do_action(Scene_t* scene, ActionType_t action, bool pre
{ {
data->scroll_area.curr_selection--; data->scroll_area.curr_selection--;
vert_scrollarea_refocus(&data->scroll_area); vert_scrollarea_refocus(&data->scroll_area);
data->update_preview = true;
} }
} }
break; break;
@ -47,6 +177,7 @@ static void level_select_do_action(Scene_t* scene, ActionType_t action, bool pre
{ {
data->scroll_area.curr_selection++; data->scroll_area.curr_selection++;
vert_scrollarea_refocus(&data->scroll_area); vert_scrollarea_refocus(&data->scroll_area);
data->update_preview = true;
} }
} }
break; break;
@ -108,6 +239,8 @@ static void level_select_do_action(Scene_t* scene, ActionType_t action, bool pre
void init_level_select_scene(LevelSelectScene_t* scene) void init_level_select_scene(LevelSelectScene_t* scene)
{ {
init_scene(&scene->scene, &level_select_do_action, 0); init_scene(&scene->scene, &level_select_do_action, 0);
scene->data.preview = LoadRenderTexture(LEVEL_PREVIEW_SIZE, LEVEL_PREVIEW_SIZE);
scene->data.update_preview = true;
add_scene_layer( add_scene_layer(
&scene->scene, scene->scene.engine->intended_window_size.x, &scene->scene, scene->scene.engine->intended_window_size.x,
scene->scene.engine->intended_window_size.y, scene->scene.engine->intended_window_size.y,
@ -150,6 +283,7 @@ void init_level_select_scene(LevelSelectScene_t* scene)
} }
ScrollAreaRenderEnd(); ScrollAreaRenderEnd();
sc_array_add(&scene->scene.systems, &level_preview_render_func);
sc_array_add(&scene->scene.systems, &level_select_render_func); sc_array_add(&scene->scene.systems, &level_select_render_func);
sc_map_put_64(&scene->scene.action_map, KEY_UP, ACTION_UP); sc_map_put_64(&scene->scene.action_map, KEY_UP, ACTION_UP);
sc_map_put_64(&scene->scene.action_map, KEY_DOWN, ACTION_DOWN); sc_map_put_64(&scene->scene.action_map, KEY_DOWN, ACTION_DOWN);
@ -159,7 +293,7 @@ void init_level_select_scene(LevelSelectScene_t* scene)
} }
void free_level_select_scene(LevelSelectScene_t* scene) void free_level_select_scene(LevelSelectScene_t* scene)
{ {
UnloadRenderTexture(scene->data.preview);
vert_scrollarea_free(&scene->data.scroll_area); vert_scrollarea_free(&scene->data.scroll_area);
free_scene(&scene->scene); free_scene(&scene->scene);
} }

View File

@ -125,7 +125,9 @@ typedef struct MenuScene {
typedef struct LevelSelectSceneData { typedef struct LevelSelectSceneData {
VertScrollArea_t scroll_area; VertScrollArea_t scroll_area;
RenderTexture2D preview;
LevelPack_t* level_pack; LevelPack_t* level_pack;
bool update_preview;
} LevelSelectSceneData_t; } LevelSelectSceneData_t;
typedef struct LevelSelectScene { typedef struct LevelSelectScene {