1
0
Fork 0
mirror of https://github.com/TerryCavanagh/VVVVVV.git synced 2025-01-08 18:09:45 +01:00

Add support for translatable sprites

Language folders can now have a graphics folder, with these files:
- sprites.png and flipsprites.png: spritesheets which contain
  translated versions of the word enemies and checkpoints
- spritesmask.xml: an XML file containing all the sprites that should
  be copied from the translated sprites and flipsprites images to
  the original sprites/flipsprites.

This means that the translated spritesheets don't have to contain ALL
sprites - they only have to contain the translated ones. When loading
them, the game assembles a combined spritesheet with translated sprites
replacing English ones as needed, and this sheet is used to visually
substitute the normal sprites at rendering time.

It's important to note that even if 32x32 enemies have pixel-perfect
hitboxes, this is only a visual change. This has been discussed several
times on Discord - basically we don't want to give people unfair
advantages or disadvantages because of their language setting, or
change existing gameplay and speedruns tactics, which may depend on the
exact pixel arrangements of the enemies. Therefore, the hitboxes are
still based on the English sprites. This should be basically
unnoticeable for casual players, especially with some thought from
translators and artists, but there will be an option in the speedrunner
menu to display the original sprites all the time.

I removed the `VVV_freefunc(SDL_FreeSurface, *tilesheet)` in
make_array() in Graphics.cpp, which frees grphx.im_sprites_surf and
grphx.im_flipsprites_surf. Since GraphicsResources::destroy() already
frees these, it looks like the only purpose the one in make_array()
serves is to do it earlier. But now we need them again later (when
switching languages) so let's just not free them early.
This commit is contained in:
Dav999 2023-09-27 03:53:36 +02:00 committed by Misa Elizabeth Kai
parent 8ef000554d
commit 9045e26d3e
6 changed files with 209 additions and 2 deletions

View file

@ -503,23 +503,82 @@ int Graphics::clear(void)
return clear(0, 0, 0, 255); return clear(0, 0, 0, 255);
} }
bool Graphics::substitute(SDL_Texture** texture)
{
/* Either keep the given texture the same and return false,
* or substitute it for a translation and return true. */
if (loc::english_sprites)
{
return false;
}
SDL_Texture* subst = NULL;
if (*texture == grphx.im_sprites)
{
subst = grphx.im_sprites_translated;
}
else if (*texture == grphx.im_flipsprites)
{
subst = grphx.im_flipsprites_translated;
}
if (subst == NULL)
{
return false;
}
// Apply the same colors as on the original
Uint8 r, g, b, a;
SDL_GetTextureColorMod(*texture, &r, &g, &b);
SDL_GetTextureAlphaMod(*texture, &a);
set_texture_color_mod(subst, r, g, b);
set_texture_alpha_mod(subst, a);
*texture = subst;
return true;
}
void Graphics::post_substitute(SDL_Texture* subst)
{
set_texture_color_mod(subst, 255, 255, 255);
set_texture_alpha_mod(subst, 255);
}
int Graphics::copy_texture(SDL_Texture* texture, const SDL_Rect* src, const SDL_Rect* dest) int Graphics::copy_texture(SDL_Texture* texture, const SDL_Rect* src, const SDL_Rect* dest)
{ {
bool is_substituted = substitute(&texture);
const int result = SDL_RenderCopy(gameScreen.m_renderer, texture, src, dest); const int result = SDL_RenderCopy(gameScreen.m_renderer, texture, src, dest);
if (result != 0) if (result != 0)
{ {
WHINE_ONCE_ARGS(("Could not copy texture: %s", SDL_GetError())); WHINE_ONCE_ARGS(("Could not copy texture: %s", SDL_GetError()));
} }
if (is_substituted)
{
post_substitute(texture);
}
return result; return result;
} }
int Graphics::copy_texture(SDL_Texture* texture, const SDL_Rect* src, const SDL_Rect* dest, const double angle, const SDL_Point* center, const SDL_RendererFlip flip) int Graphics::copy_texture(SDL_Texture* texture, const SDL_Rect* src, const SDL_Rect* dest, const double angle, const SDL_Point* center, const SDL_RendererFlip flip)
{ {
bool is_substituted = substitute(&texture);
const int result = SDL_RenderCopyEx(gameScreen.m_renderer, texture, src, dest, angle, center, flip); const int result = SDL_RenderCopyEx(gameScreen.m_renderer, texture, src, dest, angle, center, flip);
if (result != 0) if (result != 0)
{ {
WHINE_ONCE_ARGS(("Could not copy texture: %s", SDL_GetError())); WHINE_ONCE_ARGS(("Could not copy texture: %s", SDL_GetError()));
} }
if (is_substituted)
{
post_substitute(texture);
}
return result; return result;
} }
@ -3423,8 +3482,6 @@ static void make_array(
vector.push_back(temp); vector.push_back(temp);
} }
} }
VVV_freefunc(SDL_FreeSurface, *tilesheet);
} }
bool Graphics::reloadresources(void) bool Graphics::reloadresources(void)

View file

@ -196,6 +196,9 @@ public:
int clear(int r, int g, int b, int a); int clear(int r, int g, int b, int a);
int clear(void); int clear(void);
bool substitute(SDL_Texture** texture);
void post_substitute(SDL_Texture* subst);
int copy_texture(SDL_Texture* texture, const SDL_Rect* src, const SDL_Rect* dest); int copy_texture(SDL_Texture* texture, const SDL_Rect* src, const SDL_Rect* dest);
int copy_texture(SDL_Texture* texture, const SDL_Rect* src, const SDL_Rect* dest, double angle, const SDL_Point* center, SDL_RendererFlip flip); int copy_texture(SDL_Texture* texture, const SDL_Rect* src, const SDL_Rect* dest, double angle, const SDL_Point* center, SDL_RendererFlip flip);

View file

@ -1,10 +1,14 @@
#include "GraphicsResources.h" #include "GraphicsResources.h"
#include <tinyxml2.h>
#include "Alloc.h" #include "Alloc.h"
#include "FileSystemUtils.h" #include "FileSystemUtils.h"
#include "GraphicsUtil.h" #include "GraphicsUtil.h"
#include "Localization.h"
#include "Vlogging.h" #include "Vlogging.h"
#include "Screen.h" #include "Screen.h"
#include "XMLUtils.h"
// Used to load PNG data // Used to load PNG data
extern "C" extern "C"
@ -260,6 +264,134 @@ static void LoadSprites(const char* filename, SDL_Texture** texture, SDL_Surface
VVV_free(data); VVV_free(data);
} }
static void LoadSpritesTranslation(
const char* filename,
tinyxml2::XMLDocument* mask,
SDL_Surface* surface_english,
SDL_Texture** texture
) {
/* Create a sprites texture for display in another language.
* surface_english is used as a base. Parts of the translation (filename)
* will replace parts of the base, as instructed in the mask XML. */
if (surface_english == NULL)
{
vlog_error("LoadSpritesTranslation: English surface is NULL!");
return;
}
// Make a copy of the English sprites, for working with
SDL_Surface* working = GetSubSurface(
surface_english,
0, 0, surface_english->w, surface_english->h
);
if (working == NULL)
{
return;
}
SDL_Surface* translated;
{
unsigned char* data;
SDL_Surface* loaded_image = LoadImageRaw(filename, &data);
translated = LoadSurfaceFromRaw(loaded_image);
VVV_freefunc(SDL_FreeSurface, loaded_image);
VVV_free(data);
}
SDL_SetSurfaceBlendMode(translated, SDL_BLENDMODE_NONE);
tinyxml2::XMLHandle hMask(mask);
tinyxml2::XMLElement* pElem;
int sprite_w = 1, sprite_h = 1;
if ((pElem = mask->FirstChildElement()) != NULL)
{
sprite_w = pElem->IntAttribute("sprite_w", 1);
sprite_h = pElem->IntAttribute("sprite_h", 1);
}
FOR_EACH_XML_ELEMENT(hMask, pElem)
{
EXPECT_ELEM(pElem, "sprite");
int x = pElem->IntAttribute("x", 0);
int y = pElem->IntAttribute("y", 0);
SDL_Rect src;
src.x = x * sprite_w;
src.y = y * sprite_h;
src.w = pElem->IntAttribute("w", 1) * sprite_w;
src.h = pElem->IntAttribute("h", 1) * sprite_h;
SDL_Rect dst;
dst.x = pElem->IntAttribute("dx", x) * sprite_w;
dst.y = pElem->IntAttribute("dy", y) * sprite_h;
SDL_BlitSurface(translated, &src, working, &dst);
}
*texture = LoadTextureFromRaw(filename, working, TEX_WHITE);
VVV_freefunc(SDL_FreeSurface, translated);
VVV_freefunc(SDL_FreeSurface, working);
}
void GraphicsResources::init_translations(void)
{
VVV_freefunc(SDL_DestroyTexture, im_sprites_translated);
VVV_freefunc(SDL_DestroyTexture, im_flipsprites_translated);
if (loc::english_sprites)
{
return;
}
const char* langcode = loc::lang.c_str();
const char* path_template = "lang/%s/graphics/%s";
char path_xml[256];
char path_sprites[256];
char path_flipsprites[256];
SDL_snprintf(path_xml, sizeof(path_xml), path_template, langcode, "spritesmask.xml");
SDL_snprintf(path_sprites, sizeof(path_sprites), path_template, langcode, "sprites.png");
SDL_snprintf(path_flipsprites, sizeof(path_flipsprites), path_template, langcode, "flipsprites.png");
/* We don't want to apply main-game translations to level-specific (custom) sprites.
* Either sprites and translations are BOTH main-game, or BOTH level-specific.
* Our pivots are the XML (it _has_ to exist for translated sprites to work) and
* graphics/sprites.png (what sense does it make to have only flipsprites). */
if (FILESYSTEM_isAssetMounted(path_xml) != FILESYSTEM_isAssetMounted("graphics/sprites.png"))
{
return;
}
tinyxml2::XMLDocument doc_mask;
if (!FILESYSTEM_loadAssetTiXml2Document(path_xml, doc_mask))
{
// Only try to load the images if the XML document exists
return;
}
if (FILESYSTEM_areAssetsInSameRealDir(path_xml, path_sprites))
{
LoadSpritesTranslation(
path_sprites,
&doc_mask,
im_sprites_surf,
&im_sprites_translated
);
}
if (FILESYSTEM_areAssetsInSameRealDir(path_xml, path_flipsprites))
{
LoadSpritesTranslation(
path_flipsprites,
&doc_mask,
im_flipsprites_surf,
&im_flipsprites_translated
);
}
}
void GraphicsResources::init(void) void GraphicsResources::init(void)
{ {
LoadVariants("graphics/tiles.png", &im_tiles, &im_tiles_white, &im_tiles_tint); LoadVariants("graphics/tiles.png", &im_tiles, &im_tiles_white, &im_tiles_tint);
@ -285,6 +417,11 @@ void GraphicsResources::init(void)
im_image10 = LoadImage("graphics/ending.png"); im_image10 = LoadImage("graphics/ending.png");
im_image11 = LoadImage("graphics/site4.png", TEX_WHITE); im_image11 = LoadImage("graphics/site4.png", TEX_WHITE);
im_sprites_translated = NULL;
im_flipsprites_translated = NULL;
init_translations();
im_image12 = SDL_CreateTexture(gameScreen.m_renderer, SDL_PIXELFORMAT_ARGB8888, SDL_TEXTUREACCESS_TARGET, 240, 180); im_image12 = SDL_CreateTexture(gameScreen.m_renderer, SDL_PIXELFORMAT_ARGB8888, SDL_TEXTUREACCESS_TARGET, 240, 180);
if (im_image12 == NULL) if (im_image12 == NULL)
@ -324,6 +461,9 @@ void GraphicsResources::destroy(void)
CLEAR(im_image10); CLEAR(im_image10);
CLEAR(im_image11); CLEAR(im_image11);
CLEAR(im_image12); CLEAR(im_image12);
CLEAR(im_sprites_translated);
CLEAR(im_flipsprites_translated);
#undef CLEAR #undef CLEAR
VVV_freefunc(SDL_FreeSurface, im_sprites_surf); VVV_freefunc(SDL_FreeSurface, im_sprites_surf);

View file

@ -16,6 +16,8 @@ public:
void init(void); void init(void);
void destroy(void); void destroy(void);
void init_translations(void);
SDL_Surface* im_sprites_surf; SDL_Surface* im_sprites_surf;
SDL_Surface* im_flipsprites_surf; SDL_Surface* im_flipsprites_surf;
@ -43,6 +45,9 @@ public:
SDL_Texture* im_image10; SDL_Texture* im_image10;
SDL_Texture* im_image11; SDL_Texture* im_image11;
SDL_Texture* im_image12; SDL_Texture* im_image12;
SDL_Texture* im_sprites_translated;
SDL_Texture* im_flipsprites_translated;
}; };
#endif /* GRAPHICSRESOURCES_H */ #endif /* GRAPHICSRESOURCES_H */

View file

@ -1121,6 +1121,7 @@ static void menuactionpress(void)
loc::lang = loc::languagelist[game.currentmenuoption].code; loc::lang = loc::languagelist[game.currentmenuoption].code;
loc::loadtext(false); loc::loadtext(false);
loc::lang_set = true; loc::lang_set = true;
graphics.grphx.init_translations();
} }
if (loc::pre_title_lang_menu) if (loc::pre_title_lang_menu)

View file

@ -177,6 +177,7 @@ void KeyPoll::Poll(void)
{ {
/* Reload language files */ /* Reload language files */
loc::loadtext(false); loc::loadtext(false);
graphics.grphx.init_translations();
music.playef(Sound_COIN); music.playef(Sound_COIN);
} }