2 Copyright (C) 2015 Aaron Suen <warr1024@gmail.com>
4 This program is free software; you can redistribute it and/or modify
5 it under the terms of the GNU Lesser General Public License as published by
6 the Free Software Foundation; either version 2.1 of the License, or
7 (at your option) any later version.
9 This program is distributed in the hope that it will be useful,
10 but WITHOUT ANY WARRANTY; without even the implied warranty of
11 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 GNU Lesser General Public License for more details.
14 You should have received a copy of the GNU Lesser General Public License along
15 with this program; if not, write to the Free Software Foundation, Inc.,
16 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
19 #include "guiscalingfilter.h"
20 #include "imagefilters.h"
23 #include "util/numeric.h"
25 #include "client/renderingengine.h"
27 /* Maintain a static cache to store the images that correspond to textures
28 * in a format that's manipulable by code. Some platforms exhibit issues
29 * converting textures back into images repeatedly, and some don't even
32 std::map<io::path, video::IImage *> g_imgCache;
34 /* Maintain a static cache of all pre-scaled textures. These need to be
35 * cleared as well when the cached images.
37 std::map<io::path, video::ITexture *> g_txrCache;
39 /* Manually insert an image into the cache, useful to avoid texture-to-image
40 * conversion whenever we can intercept it.
42 void guiScalingCache(const io::path &key, video::IVideoDriver *driver, video::IImage *value)
44 if (!g_settings->getBool("gui_scaling_filter"))
47 if (g_imgCache.find(key) != g_imgCache.end())
48 return; // Already cached.
50 video::IImage *copied = driver->createImage(value->getColorFormat(),
51 value->getDimension());
52 value->copyTo(copied);
53 g_imgCache[key] = copied;
56 // Manually clear the cache, e.g. when switching to different worlds.
57 void guiScalingCacheClear()
59 for (auto &it : g_imgCache) {
64 for (auto &it : g_txrCache) {
66 RenderingEngine::get_video_driver()->removeTexture(it.second);
71 /* Get a cached, high-quality pre-scaled texture for display purposes. If the
72 * texture is not already cached, attempt to create it. Returns a pre-scaled texture,
73 * or the original texture if unable to pre-scale it.
75 video::ITexture *guiScalingResizeCached(video::IVideoDriver *driver,
76 video::ITexture *src, const core::rect<s32> &srcrect,
77 const core::rect<s32> &destrect)
81 if (!g_settings->getBool("gui_scaling_filter"))
84 // Calculate scaled texture name.
86 porting::mt_snprintf(rectstr, sizeof(rectstr), "%d:%d:%d:%d:%d:%d",
87 srcrect.UpperLeftCorner.X,
88 srcrect.UpperLeftCorner.Y,
92 destrect.getHeight());
93 io::path origname = src->getName().getPath();
94 io::path scalename = origname + "@guiScalingFilter:" + rectstr;
96 // Search for existing scaled texture.
97 auto it_txr = g_txrCache.find(scalename);
98 video::ITexture *scaled = (it_txr != g_txrCache.end()) ? it_txr->second : nullptr;
102 // Try to find the texture converted to an image in the cache.
103 // If the image was not found, try to extract it from the texture.
104 auto it_img = g_imgCache.find(origname);
105 video::IImage *srcimg = (it_img != g_imgCache.end()) ? it_img->second : nullptr;
107 if (!g_settings->getBool("gui_scaling_filter_txr2img"))
109 srcimg = driver->createImageFromData(src->getColorFormat(),
110 src->getSize(), src->lock(video::ETLM_READ_ONLY), false);
112 g_imgCache[origname] = srcimg;
115 // Create a new destination image and scale the source into it.
116 imageCleanTransparent(srcimg, 0);
117 video::IImage *destimg = driver->createImage(src->getColorFormat(),
118 core::dimension2d<u32>((u32)destrect.getWidth(),
119 (u32)destrect.getHeight()));
120 imageScaleNNAA(srcimg, srcrect, destimg);
123 // Some platforms are picky about textures being powers of 2, so expand
124 // the image dimensions to the next power of 2, if necessary.
125 if (!driver->queryFeature(video::EVDF_TEXTURE_NPOT)) {
126 video::IImage *po2img = driver->createImage(src->getColorFormat(),
127 core::dimension2d<u32>(npot2((u32)destrect.getWidth()),
128 npot2((u32)destrect.getHeight())));
129 po2img->fill(video::SColor(0, 0, 0, 0));
130 destimg->copyTo(po2img);
136 // Convert the scaled image back into a texture.
137 scaled = driver->addTexture(scalename, destimg);
139 g_txrCache[scalename] = scaled;
144 /* Convenience wrapper for guiScalingResizeCached that accepts parameters that
145 * are available at GUI imagebutton creation time.
147 video::ITexture *guiScalingImageButton(video::IVideoDriver *driver,
148 video::ITexture *src, s32 width, s32 height)
152 return guiScalingResizeCached(driver, src,
153 core::rect<s32>(0, 0, src->getSize().Width, src->getSize().Height),
154 core::rect<s32>(0, 0, width, height));
157 /* Replacement for driver->draw2DImage() that uses the high-quality pre-scaled
158 * texture, if configured.
160 void draw2DImageFilterScaled(video::IVideoDriver *driver, video::ITexture *txr,
161 const core::rect<s32> &destrect, const core::rect<s32> &srcrect,
162 const core::rect<s32> *cliprect, const video::SColor *const colors,
165 // Attempt to pre-scale image in software in high quality.
166 video::ITexture *scaled = guiScalingResizeCached(driver, txr, srcrect, destrect);
170 // Correct source rect based on scaled image.
171 const core::rect<s32> mysrcrect = (scaled != txr)
172 ? core::rect<s32>(0, 0, destrect.getWidth(), destrect.getHeight())
175 driver->draw2DImage(scaled, destrect, mysrcrect, cliprect, colors, usealpha);
178 void draw2DImage9Slice(video::IVideoDriver *driver, video::ITexture *texture,
179 const core::rect<s32> &rect, const core::rect<s32> &middle,
180 const core::rect<s32> *cliprect, const video::SColor *const colors)
182 auto originalSize = texture->getOriginalSize();
183 core::vector2di lowerRightOffset = core::vector2di(originalSize.Width, originalSize.Height) - middle.LowerRightCorner;
185 for (int y = 0; y < 3; ++y) {
186 for (int x = 0; x < 3; ++x) {
187 core::rect<s32> src({0, 0}, originalSize);
188 core::rect<s32> dest = rect;
192 dest.LowerRightCorner.X = rect.UpperLeftCorner.X + middle.UpperLeftCorner.X;
193 src.LowerRightCorner.X = middle.UpperLeftCorner.X;
197 dest.UpperLeftCorner.X += middle.UpperLeftCorner.X;
198 dest.LowerRightCorner.X -= lowerRightOffset.X;
199 src.UpperLeftCorner.X = middle.UpperLeftCorner.X;
200 src.LowerRightCorner.X = middle.LowerRightCorner.X;
204 dest.UpperLeftCorner.X = rect.LowerRightCorner.X - lowerRightOffset.X;
205 src.UpperLeftCorner.X = middle.LowerRightCorner.X;
211 dest.LowerRightCorner.Y = rect.UpperLeftCorner.Y + middle.UpperLeftCorner.Y;
212 src.LowerRightCorner.Y = middle.UpperLeftCorner.Y;
216 dest.UpperLeftCorner.Y += middle.UpperLeftCorner.Y;
217 dest.LowerRightCorner.Y -= lowerRightOffset.Y;
218 src.UpperLeftCorner.Y = middle.UpperLeftCorner.Y;
219 src.LowerRightCorner.Y = middle.LowerRightCorner.Y;
223 dest.UpperLeftCorner.Y = rect.LowerRightCorner.Y - lowerRightOffset.Y;
224 src.UpperLeftCorner.Y = middle.LowerRightCorner.Y;
228 draw2DImageFilterScaled(driver, texture, dest, src, cliprect, colors, true);