#include #include #include "Internal.h" #include "modules/DFSDL.h" #include "modules/Textures.h" #include "Debug.h" #include "PluginManager.h" #include "VTableInterpose.h" #include "df/enabler.h" #include "df/viewscreen_adopt_regionst.h" #include "df/viewscreen_loadgamest.h" #include "df/viewscreen_new_regionst.h" #include "df/viewscreen_new_arenast.h" using df::global::enabler; using namespace DFHack; using namespace DFHack::DFSDL; namespace DFHack { DBG_DECLARE(core, textures, DebugCategory::LINFO); } static bool g_loaded = false; static long g_num_dfhack_textures = 0; static long g_dfhack_logo_texpos_start = -1; static long g_green_pin_texpos_start = -1; static long g_red_pin_texpos_start = -1; static long g_icons_texpos_start = -1; static long g_on_off_texpos_start = -1; static long g_pathable_texpos_start = -1; static long g_unsuspend_texpos_start = -1; static long g_control_panel_texpos_start = -1; static long g_thin_borders_texpos_start = -1; static long g_medium_borders_texpos_start = -1; static long g_bold_borders_texpos_start = -1; static long g_panel_borders_texpos_start = -1; static long g_window_borders_texpos_start = -1; static std::unordered_map g_handle_to_texpos; static std::unordered_map g_handle_to_surface; static std::mutex g_adding_mutex; const uint32_t TILE_WIDTH_PX = 8; const uint32_t TILE_HEIGHT_PX = 12; // Converts an arbitrary Surface to something like the display format // (32-bit RGBA), and converts magenta to transparency if convert_magenta is set // and the source surface didn't already have an alpha channel. // It also deletes the source surface. // // It uses the same pixel format (RGBA, R at lowest address) regardless of // hardware. SDL_Surface* canonicalize_format(SDL_Surface* src) { SDL_PixelFormat fmt; fmt.palette = NULL; fmt.BitsPerPixel = 32; fmt.BytesPerPixel = 4; fmt.Rloss = fmt.Gloss = fmt.Bloss = fmt.Aloss = 0; #if SDL_BYTEORDER == SDL_BIG_ENDIAN fmt.Rshift = 24; fmt.Gshift = 16; fmt.Bshift = 8; fmt.Ashift = 0; #else fmt.Rshift = 0; fmt.Gshift = 8; fmt.Bshift = 16; fmt.Ashift = 24; #endif fmt.Rmask = 255 << fmt.Rshift; fmt.Gmask = 255 << fmt.Gshift; fmt.Bmask = 255 << fmt.Bshift; fmt.Amask = 255 << fmt.Ashift; SDL_Surface* tgt = DFSDL_ConvertSurface(src, &fmt, SDL_SWSURFACE); DFSDL_FreeSurface(src); for (int x = 0; x < tgt->w; ++x) { for (int y = 0; y < tgt->h; ++y) { Uint8* p = (Uint8*)tgt->pixels + y * tgt->pitch + x * 4; if (p[3] == 0) { for (int c = 0; c < 3; c++) { p[c] = 0; } } } } return tgt; } // register surface in texture raws, get a texpos static long add_texture(SDL_Surface* surface) { std::lock_guard lg_add_texture(g_adding_mutex); auto texpos = enabler->textures.raws.size(); enabler->textures.raws.push_back(surface); return texpos; } TexposHandle Textures::loadTexture(SDL_Surface* surface) { if (!surface) return 0; // should be some error, i guess auto handle = reinterpret_cast(surface); g_handle_to_surface.emplace(handle, surface); surface->refcount++; // prevent destruct on next FreeSurface by game auto texpos = add_texture(surface); g_handle_to_texpos.emplace(handle, texpos); return handle; } std::vector Textures::loadTileset(const std::string& file, int tile_px_w = TILE_WIDTH_PX, int tile_px_h = TILE_HEIGHT_PX) { std::vector handles{}; SDL_Surface* surface = DFIMG_Load(file.c_str()); if (!surface) { ERR(textures).printerr("unable to load textures from '%s'\n", file.c_str()); return handles; } surface = canonicalize_format(surface); int dimx = surface->w / tile_px_w; int dimy = surface->h / tile_px_h; for (int y = 0; y < dimy; y++) { for (int x = 0; x < dimx; x++) { SDL_Surface* tile = DFSDL_CreateRGBSurface(0, tile_px_w, tile_px_h, 32, surface->format->Rmask, surface->format->Gmask, surface->format->Bmask, surface->format->Amask); SDL_Rect vp{ tile_px_w * x, tile_px_h * y, tile_px_w, tile_px_h }; DFSDL_UpperBlit(surface, &vp, tile, NULL); auto handle = Textures::loadTexture(tile); handles.push_back(handle); } } DFSDL_FreeSurface(surface); DEBUG(textures).print("loaded %ld textures from '%s'\n", handles.size(), file.c_str()); return handles; } long Textures::getTexposByHandle(TexposHandle handle) { if (!handle) return -1; if (g_handle_to_texpos.contains(handle)) return g_handle_to_texpos[handle]; if (g_handle_to_surface.contains(handle)) { g_handle_to_surface[handle]->refcount++; // prevent destruct on next FreeSurface by game auto texpos = add_texture(g_handle_to_surface[handle]); g_handle_to_texpos.emplace(handle, texpos); return texpos; } return -1; } static void reset_texpos() { g_handle_to_texpos.clear(); } static void reset_surface() { for (auto& entry : g_handle_to_surface) { DFSDL_FreeSurface(entry.second); } g_handle_to_surface.clear(); } // reset point on New Game struct tracking_stage_new_region : df::viewscreen_new_regionst { typedef df::viewscreen_new_regionst interpose_base; DEFINE_VMETHOD_INTERPOSE(void, logic, ()) { if (this->m_raw_load_stage != this->raw_load_stage) { this->m_raw_load_stage = this->raw_load_stage; if (this->m_raw_load_stage == 1) reset_texpos(); } INTERPOSE_NEXT(logic)(); } private: inline static int m_raw_load_stage = -2; // not valid state at the start }; IMPLEMENT_VMETHOD_INTERPOSE(tracking_stage_new_region, logic); // reset point on New Game in Existing World struct tracking_stage_adopt_region : df::viewscreen_adopt_regionst { typedef df::viewscreen_adopt_regionst interpose_base; DEFINE_VMETHOD_INTERPOSE(void, logic, ()) { if (this->m_cur_step != this->cur_step) { this->m_cur_step = this->cur_step; if (this->m_cur_step == df::viewscreen_adopt_regionst::T_cur_step::ProcessingRawData) reset_texpos(); } INTERPOSE_NEXT(logic)(); } private: inline static int m_cur_step = -2; // not valid state at the start }; IMPLEMENT_VMETHOD_INTERPOSE(tracking_stage_adopt_region, logic); // reset point on Load Game struct tracking_stage_load_region : df::viewscreen_loadgamest { typedef df::viewscreen_loadgamest interpose_base; DEFINE_VMETHOD_INTERPOSE(void, logic, ()) { if (this->m_cur_step != this->cur_step) { this->m_cur_step = this->cur_step; if (this->m_cur_step == df::viewscreen_loadgamest::T_cur_step::ProcessingRawData) reset_texpos(); } INTERPOSE_NEXT(logic)(); } private: inline static int m_cur_step = -2; // not valid state at the start }; IMPLEMENT_VMETHOD_INTERPOSE(tracking_stage_load_region, logic); // reset point on New Arena struct tracking_stage_new_arena : df::viewscreen_new_arenast { typedef df::viewscreen_new_arenast interpose_base; DEFINE_VMETHOD_INTERPOSE(void, logic, ()) { if (this->m_cur_step != this->cur_step) { this->m_cur_step = this->cur_step; if (this->m_cur_step == 0) reset_texpos(); } INTERPOSE_NEXT(logic)(); } private: inline static int m_cur_step = -2; // not valid state at the start }; IMPLEMENT_VMETHOD_INTERPOSE(tracking_stage_new_arena, logic); static void install_reset_point() { INTERPOSE_HOOK(tracking_stage_new_region, logic).apply(); INTERPOSE_HOOK(tracking_stage_adopt_region, logic).apply(); INTERPOSE_HOOK(tracking_stage_load_region, logic).apply(); INTERPOSE_HOOK(tracking_stage_new_arena, logic).apply(); } static void uninstall_reset_point() { INTERPOSE_HOOK(tracking_stage_new_region, logic).remove(); INTERPOSE_HOOK(tracking_stage_adopt_region, logic).remove(); INTERPOSE_HOOK(tracking_stage_load_region, logic).remove(); INTERPOSE_HOOK(tracking_stage_new_arena, logic).remove(); } static size_t load_tiles_from_image(color_ostream& out, const char* fname, long* texpos_start, int tile_w = TILE_WIDTH_PX, int tile_h = TILE_HEIGHT_PX) { SDL_Surface* s = DFIMG_Load(fname); if (!s) { out.printerr("unable to load textures from '%s'\n", fname); return 0; } s = canonicalize_format(s); int dimx = s->w / tile_w; int dimy = s->h / tile_h; long count = 0; for (int y = 0; y < dimy; y++) { for (int x = 0; x < dimx; x++) { SDL_Surface* tile = DFSDL_CreateRGBSurface(0, // SDL_SWSURFACE tile_w, tile_h, 32, s->format->Rmask, s->format->Gmask, s->format->Bmask, s->format->Amask); SDL_Rect vp; vp.x = tile_w * x; vp.y = tile_h * y; vp.w = tile_w; vp.h = tile_h; DFSDL_UpperBlit(s, &vp, tile, NULL); if (!count++) *texpos_start = enabler->textures.raws.size(); enabler->textures.raws.push_back(tile); } } DFSDL_FreeSurface(s); DEBUG(textures,out).print("loaded %ld textures from '%s'\n", count, fname); return count; } void Textures::initDynamic(color_ostream& out) { install_reset_point(); DEBUG(textures, out).print("dynamic texture reset points installed"); } // DFHack could conceivably be loaded at any time, so we need to be able to // handle loading textures before or after a world is loaded. // If a world is already loaded, then append our textures to the raws. they'll // be freed when the world is unloaded and we'll reload when we get to the title // screen. If it's pre-world, append our textures and then adjust the "init" // texture count so our textures will no longer be freed when worlds are // unloaded. // void Textures::init(color_ostream& out) { if (!enabler) return; auto& textures = enabler->textures; long num_textures = textures.raws.size(); if (num_textures <= g_dfhack_logo_texpos_start) g_loaded = false; if (g_loaded) return; bool is_pre_world = num_textures == textures.init_texture_size; g_num_dfhack_textures = load_tiles_from_image(out, "hack/data/art/dfhack.png", &g_dfhack_logo_texpos_start); g_num_dfhack_textures += load_tiles_from_image(out, "hack/data/art/green-pin.png", &g_green_pin_texpos_start); g_num_dfhack_textures += load_tiles_from_image(out, "hack/data/art/red-pin.png", &g_red_pin_texpos_start); g_num_dfhack_textures += load_tiles_from_image(out, "hack/data/art/icons.png", &g_icons_texpos_start); g_num_dfhack_textures += load_tiles_from_image(out, "hack/data/art/on-off.png", &g_on_off_texpos_start); g_num_dfhack_textures += load_tiles_from_image(out, "hack/data/art/pathable.png", &g_pathable_texpos_start, 32, 32); g_num_dfhack_textures += load_tiles_from_image(out, "hack/data/art/unsuspend.png", &g_unsuspend_texpos_start, 32, 32); g_num_dfhack_textures += load_tiles_from_image(out, "hack/data/art/control-panel.png", &g_control_panel_texpos_start); g_num_dfhack_textures += load_tiles_from_image(out, "hack/data/art/border-thin.png", &g_thin_borders_texpos_start); g_num_dfhack_textures += load_tiles_from_image(out, "hack/data/art/border-medium.png", &g_medium_borders_texpos_start); g_num_dfhack_textures += load_tiles_from_image(out, "hack/data/art/border-bold.png", &g_bold_borders_texpos_start); g_num_dfhack_textures += load_tiles_from_image(out, "hack/data/art/border-panel.png", &g_panel_borders_texpos_start); g_num_dfhack_textures += load_tiles_from_image(out, "hack/data/art/border-window.png", &g_window_borders_texpos_start); DEBUG(textures,out).print("loaded %ld textures\n", g_num_dfhack_textures); if (is_pre_world) textures.init_texture_size += g_num_dfhack_textures; // NOTE: when GL modes are supported, we'll have to re-upload textures here g_loaded = true; } // It's ok to leave NULLs in the raws list (according to usage in g_src) void Textures::cleanup() { reset_texpos(); reset_surface(); uninstall_reset_point(); if (!g_loaded) return; auto& textures = enabler->textures; auto& raws = textures.raws; size_t texpos_end = g_dfhack_logo_texpos_start + g_num_dfhack_textures - 1; for (size_t idx = g_dfhack_logo_texpos_start; idx <= texpos_end; ++idx) { DFSDL_FreeSurface((SDL_Surface *)raws[idx]); raws[idx] = NULL; } if (g_dfhack_logo_texpos_start == textures.init_texture_size - g_num_dfhack_textures) textures.init_texture_size -= g_num_dfhack_textures; g_loaded = false; g_num_dfhack_textures = 0; g_dfhack_logo_texpos_start = -1; } long Textures::getDfhackLogoTexposStart() { return g_dfhack_logo_texpos_start; } long Textures::getGreenPinTexposStart() { return g_green_pin_texpos_start; } long Textures::getRedPinTexposStart() { return g_red_pin_texpos_start; } long Textures::getIconsTexposStart() { return g_icons_texpos_start; } long Textures::getOnOffTexposStart() { return g_on_off_texpos_start; } long Textures::getMapPathableTexposStart() { return g_pathable_texpos_start; } long Textures::getMapUnsuspendTexposStart() { return g_unsuspend_texpos_start; } long Textures::getControlPanelTexposStart() { return g_control_panel_texpos_start; } long Textures::getThinBordersTexposStart() { return g_thin_borders_texpos_start; } long Textures::getMediumBordersTexposStart() { return g_medium_borders_texpos_start; } long Textures::getBoldBordersTexposStart() { return g_bold_borders_texpos_start; } long Textures::getPanelBordersTexposStart() { return g_panel_borders_texpos_start; } long Textures::getWindowBordersTexposStart() { return g_window_borders_texpos_start; }