From 8f38d5fe66ca6c05b969decee252af5a9611a6cc Mon Sep 17 00:00:00 2001 From: Tim Felgentreff <timfelgentreff@gmail.com> Date: Tue, 26 Apr 2022 21:38:37 +0200 Subject: [PATCH] clean up MakeShadow function a bit --- src/include/video.h | 2 +- src/video/graphic.cpp | 130 ++++++++++++++++++------------------------ 2 files changed, 58 insertions(+), 74 deletions(-) diff --git a/src/include/video.h b/src/include/video.h index 9d081f401..132ee2bf0 100644 --- a/src/include/video.h +++ b/src/include/video.h @@ -90,7 +90,7 @@ public: class CGraphic : public gcn::Image { - +public: struct frame_pos_t { short int x; short int y; diff --git a/src/video/graphic.cpp b/src/video/graphic.cpp index 0549de338..bdebc8e4c 100644 --- a/src/video/graphic.cpp +++ b/src/video/graphic.cpp @@ -1149,6 +1149,56 @@ static inline void dither(SDL_Surface *Surface) { } } +static void applyAlphaGrayscaleToSurface(SDL_Surface **src, int alpha) +{ + SDL_Surface *alphaSurface = SDL_CreateRGBSurface(0, (*src)->w, (*src)->h, 32, RMASK, GMASK, BMASK, AMASK); + SDL_BlitSurface(*src, NULL, alphaSurface, NULL); + SDL_SetSurfaceAlphaMod(alphaSurface, alpha); + SDL_SetSurfaceColorMod(alphaSurface, 0, 0, 0); + SDL_FreeSurface(*src); + *src = alphaSurface; +} + +static void shrinkSurfaceFramesInY(SDL_Surface **src, int shrink, int numFrames, CGraphic::frame_pos_t *frameMap, int frameW, int frameH) +{ + shrink = std::abs(shrink); + SDL_Surface *alphaSurface = SDL_CreateRGBSurface(0, (*src)->w, (*src)->h, 32, RMASK, GMASK, BMASK, AMASK); + for (int f = 0; f < numFrames; f++) { + int frameX = frameMap[f].x; + int frameY = frameMap[f].y; + const SDL_Rect srcRect = { frameX, frameY, frameW, frameH }; + SDL_Rect dstRect = { frameX, frameY + shrink / 2, frameW, frameH - (shrink - shrink / 2) }; + SDL_BlitScaled(*src, &srcRect, alphaSurface, &dstRect); + } + SDL_FreeSurface(*src); + *src = alphaSurface; +} + +static void shearSurface(SDL_Surface *surface, int xOffset, int yOffset, int numFrames, CGraphic::frame_pos_t *frameMap, int frameW, int frameH) +{ + if (yOffset || xOffset) { + SDL_LockSurface(surface); + uint32_t* pixels = (uint32_t *)surface->pixels; + int pitch = surface->pitch / sizeof(uint32_t); + for (int f = 0; f < numFrames; f++) { + int frameX = frameMap[f].x; + int frameY = frameMap[f].y; + for (int x = xOffset > 0 ? 0 : frameW - 1; xOffset > 0 ? x < frameW : x >= 0; xOffset > 0 ? x++ : x--) { + for (int y = yOffset > 0 ? 0 : frameH - 1; yOffset > 0 ? y < frameH : y >= 0; yOffset > 0 ? y++ : y--) { + int xNew = x + xOffset * y / frameH; + int yNew = y + yOffset * x / frameW; + if (xNew < 0 || yNew < 0 || xNew >= frameW || yNew >= frameH) { + pixels[x + frameX + (y + frameY) * pitch] = 0; + } else { + pixels[x + frameX + (y + frameY) * pitch] = pixels[xNew + frameX + (yNew + frameY) * pitch]; + } + } + } + } + SDL_UnlockSurface(surface); + } +} + /** ** Make shadow sprite ** @@ -1157,20 +1207,10 @@ static inline void dither(SDL_Surface *Surface) { void CGraphic::MakeShadow(int xOffset, int yOffset) { VideoPaletteListRemove(Surface); - SDL_Surface *alphaSurface = SDL_CreateRGBSurface(0, Surface->w, Surface->h, 32, RMASK, GMASK, BMASK, AMASK); - SDL_BlitSurface(Surface, NULL, alphaSurface, NULL); - SDL_SetSurfaceAlphaMod(alphaSurface, 80); - SDL_SetSurfaceColorMod(alphaSurface, 0, 0, 0); - SDL_FreeSurface(Surface); - Surface = alphaSurface; + applyAlphaGrayscaleToSurface(&Surface, 80); if (SurfaceFlip) { VideoPaletteListRemove(SurfaceFlip); - alphaSurface = SDL_CreateRGBSurface(0, SurfaceFlip->w, SurfaceFlip->h, 32, RMASK, GMASK, BMASK, AMASK); - SDL_BlitSurface(SurfaceFlip, NULL, alphaSurface, NULL); - SDL_SetSurfaceAlphaMod(alphaSurface, 80); - SDL_SetSurfaceColorMod(alphaSurface, 0, 0, 0); - SDL_FreeSurface(SurfaceFlip); - SurfaceFlip = alphaSurface; + applyAlphaGrayscaleToSurface(&SurfaceFlip, 80); } // BEGIN HACK: XXX: FIXME: positive yOffset is used for fliers for now, these should not get shearing. @@ -1183,71 +1223,15 @@ void CGraphic::MakeShadow(int xOffset, int yOffset) // The sun shines from the same angle on to both normal and flipped sprites :) // 1. Shrink each frame in y-direction based on the x offset - alphaSurface = SDL_CreateRGBSurface(0, Surface->w, Surface->h, 32, RMASK, GMASK, BMASK, AMASK); - for (int f = 0; f < NumFrames; f++) { - int frameX = frame_map[f].x; - int frameY = frame_map[f].y; - const SDL_Rect srcRect = { frameX, frameY, Width, Height }; - SDL_Rect dstRect = { frameX, frameY + std::abs(xOffset) / 2, Width, Height - (std::abs(xOffset) - std::abs(xOffset) / 2) }; - SDL_BlitScaled(Surface, &srcRect, alphaSurface, &dstRect); - } - SDL_FreeSurface(Surface); - Surface = alphaSurface; + shrinkSurfaceFramesInY(&Surface, xOffset, NumFrames, frame_map, Width, Height); if (SurfaceFlip) { - alphaSurface = SDL_CreateRGBSurface(0, SurfaceFlip->w, SurfaceFlip->h, 32, RMASK, GMASK, BMASK, AMASK); - for (int f = 0; f < NumFrames; f++) { - int frameX = frameFlip_map[f].x; - int frameY = frameFlip_map[f].y; - const SDL_Rect srcRect = { frameX, frameY, Width, Height }; - SDL_Rect dstRect = { frameX, frameY + std::abs(xOffset) / 2, Width, Height - (std::abs(xOffset) - std::abs(xOffset) / 2) }; - SDL_BlitScaled(SurfaceFlip, &srcRect, alphaSurface, &dstRect); - } - SDL_FreeSurface(SurfaceFlip); - SurfaceFlip = alphaSurface; + shrinkSurfaceFramesInY(&SurfaceFlip, xOffset, NumFrames, frameFlip_map, Width, Height); } // 2. Apply shearing - if (yOffset || xOffset) { - SDL_LockSurface(Surface); - uint32_t* pixels = (uint32_t *)Surface->pixels; - int pitch = Surface->pitch / sizeof(uint32_t); - for (int f = 0; f < NumFrames; f++) { - int frameX = frame_map[f].x; - int frameY = frame_map[f].y; - for (int x = xOffset > 0 ? 0 : Width - 1; xOffset > 0 ? x < Width : x >= 0; xOffset > 0 ? x++ : x--) { - for (int y = yOffset > 0 ? 0 : Height - 1; yOffset > 0 ? y < Height : y >= 0; yOffset > 0 ? y++ : y--) { - int xNew = x + xOffset * y / Height; - int yNew = y + yOffset * x / Width; - if (xNew < 0 || yNew < 0 || xNew >= Width || yNew >= Height) { - pixels[x + frameX + (y + frameY) * pitch] = 0; - } else { - pixels[x + frameX + (y + frameY) * pitch] = pixels[xNew + frameX + (yNew + frameY) * pitch]; - } - } - } - } - SDL_UnlockSurface(Surface); - } - if ((yOffset || xOffset) && SurfaceFlip) { - SDL_LockSurface(SurfaceFlip); - uint32_t* pixels = (uint32_t *)SurfaceFlip->pixels; - int pitch = SurfaceFlip->pitch / sizeof(uint32_t); - for (int f = 0; f < NumFrames; f++) { - int frameX = frameFlip_map[f].x; - int frameY = frameFlip_map[f].y; - for (int x = xOffset > 0 ? 0 : Width - 1; xOffset > 0 ? x < Width : x >= 0; xOffset > 0 ? x++ : x--) { - for (int y = yOffset > 0 ? 0 : Height - 1; yOffset > 0 ? y < Height : y >= 0; yOffset > 0 ? y++ : y--) { - int xNew = x + xOffset * y / Height; - int yNew = y + yOffset * x / Width; - if (xNew < 0 || yNew < 0 || xNew >= Width || yNew >= Height) { - pixels[x + frameX + (y + frameY) * pitch] = 0; - } else { - pixels[x + frameX + (y + frameY) * pitch] = pixels[xNew + frameX + (yNew + frameY) * pitch]; - } - } - } - } - SDL_UnlockSurface(SurfaceFlip); + shearSurface(Surface, xOffset, yOffset, NumFrames, frame_map, Width, Height); + if (SurfaceFlip) { + shearSurface(SurfaceFlip, xOffset, yOffset, NumFrames, frameFlip_map, Width, Height); } }