| 
									
										
										
										
											2025-06-22 21:05:09 +02:00
										 |  |  | /*
 | 
					
						
							|  |  |  | MIT License | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Copyright (c) 2024 - 2025 tobid7 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Permission is hereby granted, free of charge, to any person obtaining a copy | 
					
						
							|  |  |  | of this software and associated documentation files (the "Software"), to deal | 
					
						
							|  |  |  | in the Software without restriction, including without limitation the rights | 
					
						
							|  |  |  | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | 
					
						
							|  |  |  | copies of the Software, and to permit persons to whom the Software is | 
					
						
							|  |  |  | furnished to do so, subject to the following conditions: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | The above copyright notice and this permission notice shall be included in all | 
					
						
							|  |  |  | copies or substantial portions of the Software. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | 
					
						
							|  |  |  | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | 
					
						
							|  |  |  | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | 
					
						
							|  |  |  | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | 
					
						
							|  |  |  | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | 
					
						
							|  |  |  | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | 
					
						
							|  |  |  | SOFTWARE. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef PD_IMAGE_BUILD_SHARED
 | 
					
						
							|  |  |  | #define STB_IMAGE_IMPLEMENTATION
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <pd/external/stb_image.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <cstring>
 | 
					
						
							|  |  |  | #include <memory>
 | 
					
						
							|  |  |  | #include <pd/image/image.hpp>
 | 
					
						
							|  |  |  | #include <pd/image/img_convert.hpp>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace PD { | 
					
						
							|  |  |  | PD_IMAGE_API void Image::Load(const std::string& path) { | 
					
						
							|  |  |  |   u8* img = stbi_load(path.c_str(), &pWidth, &pHeight, &fmt, 4); | 
					
						
							|  |  |  |   if (fmt == 3) { | 
					
						
							|  |  |  |     stbi_image_free(img); | 
					
						
							|  |  |  |     img = stbi_load(path.c_str(), &pWidth, &pHeight, &fmt, 3); | 
					
						
							|  |  |  |     pBuffer = std::vector<PD::u8>(img, img + (pWidth * pHeight * 3)); | 
					
						
							|  |  |  |     pFmt = RGB; | 
					
						
							|  |  |  |     stbi_image_free(img); | 
					
						
							|  |  |  |   } else if (fmt == 4) { | 
					
						
							|  |  |  |     pBuffer = std::vector<PD::u8>(img, img + (pWidth * pHeight * 4)); | 
					
						
							|  |  |  |     pFmt = RGBA; | 
					
						
							|  |  |  |     stbi_image_free(img); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | PD_IMAGE_API void Image::Load(const std::vector<u8>& buf) { | 
					
						
							|  |  |  |   u8* img = | 
					
						
							|  |  |  |       stbi_load_from_memory(buf.data(), buf.size(), &pWidth, &pHeight, &fmt, 4); | 
					
						
							|  |  |  |   if (fmt == 3) { | 
					
						
							|  |  |  |     stbi_image_free(img); | 
					
						
							|  |  |  |     img = stbi_load_from_memory(buf.data(), buf.size(), &pWidth, &pHeight, &fmt, | 
					
						
							|  |  |  |                                 3); | 
					
						
							|  |  |  |     pBuffer = std::vector<PD::u8>(img, img + (pWidth * pHeight * 3)); | 
					
						
							|  |  |  |     pFmt = RGB; | 
					
						
							|  |  |  |     stbi_image_free(img); | 
					
						
							|  |  |  |   } else if (fmt == 4) { | 
					
						
							|  |  |  |     pBuffer = std::vector<PD::u8>(img, img + (pWidth * pHeight * 4)); | 
					
						
							|  |  |  |     stbi_image_free(img); | 
					
						
							|  |  |  |     pFmt = RGBA; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | PD_IMAGE_API void Image::Copy(const std::vector<u8>& buf, int w, int h, | 
					
						
							|  |  |  |                               int bpp) { | 
					
						
							|  |  |  |   this->fmt = bpp; | 
					
						
							|  |  |  |   if (buf.size() != (size_t)w * h * bpp) { | 
					
						
							|  |  |  |     // Size Error
 | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   this->pBuffer.resize(w * h * bpp); | 
					
						
							|  |  |  |   for (size_t i = 0; i < this->pBuffer.size(); i++) { | 
					
						
							|  |  |  |     pBuffer[i] = buf[i]; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-07-23 23:21:34 +02:00
										 |  |  | PD_IMAGE_API void Image::FlipHorizontal() { | 
					
						
							|  |  |  |   /**
 | 
					
						
							|  |  |  |    * Dont know if i am brain dead but i think this code | 
					
						
							|  |  |  |    * should Horizpntal flip an image | 
					
						
							|  |  |  |    * Probably this needs some optimisation like not always calling | 
					
						
							|  |  |  |    * Fmt2Bpp and use `* 0.5` instead of `/ 2` i guess | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   for (int i = 0; i < pWidth / 2; i++) { | 
					
						
							|  |  |  |     for (int j = 0; j < pHeight; j++) { | 
					
						
							|  |  |  |       int src = (j * pWidth + i) * Fmt2Bpp(pFmt); | 
					
						
							|  |  |  |       int dst = (j * pWidth + (pWidth - 1 - i)) * Fmt2Bpp(pFmt); | 
					
						
							|  |  |  |       for (int k = 0; k < Fmt2Bpp(pFmt); k++) { | 
					
						
							|  |  |  |         PD::u8 tmp = pBuffer[dst + k]; | 
					
						
							|  |  |  |         pBuffer[dst + k] = pBuffer[src + k]; | 
					
						
							|  |  |  |         pBuffer[src + k] = tmp; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | PD_IMAGE_API void Image::FlipVertical() { | 
					
						
							|  |  |  |   /**
 | 
					
						
							|  |  |  |    * Dont know if i am brain dead but i think this code | 
					
						
							|  |  |  |    * should Vertical flip an image | 
					
						
							|  |  |  |    * Probably this needs some optimisation like not always calling | 
					
						
							|  |  |  |    * Fmt2Bpp and use `* 0.5` instead of `/ 2` i guess | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   for (int i = 0; i < pWidth; i++) { | 
					
						
							|  |  |  |     for (int j = 0; j < pHeight / 2; j++) { | 
					
						
							|  |  |  |       int src = (j * pWidth + i) * Fmt2Bpp(pFmt); | 
					
						
							|  |  |  |       int dst = ((pHeight - 1 - j) * pWidth + i) * Fmt2Bpp(pFmt); | 
					
						
							|  |  |  |       for (int k = 0; k < Fmt2Bpp(pFmt); k++) { | 
					
						
							|  |  |  |         PD::u8 tmp = pBuffer[dst + k]; | 
					
						
							|  |  |  |         pBuffer[dst + k] = pBuffer[src + k]; | 
					
						
							|  |  |  |         pBuffer[src + k] = tmp; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-22 21:05:09 +02:00
										 |  |  | PD_IMAGE_API void Image::Convert(Image::Ref img, Image::Format dst) { | 
					
						
							|  |  |  |   if (img->pFmt == dst) { | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } else if (img->pFmt == Image::RGB && dst == Image::BGR) { | 
					
						
							|  |  |  |     ImgConvert::ReverseBuf(img->pBuffer, 3, img->pWidth, img->pHeight); | 
					
						
							|  |  |  |     img->pFmt = BGR; | 
					
						
							|  |  |  |   } else if (img->pFmt == Image::RGB && dst == Image::RGBA) { | 
					
						
							|  |  |  |     std::vector<PD::u8> cpy = img->pBuffer; | 
					
						
							|  |  |  |     img->pBuffer.resize(img->pWidth * img->pHeight * 4); | 
					
						
							|  |  |  |     ImgConvert::RGB24toRGBA32(img->pBuffer, cpy, img->pWidth, img->pHeight); | 
					
						
							|  |  |  |     img->pFmt = RGBA; | 
					
						
							|  |  |  |   } else if (img->pFmt == Image::RGBA && dst == Image::RGB) { | 
					
						
							|  |  |  |     std::vector<PD::u8> cpy = img->pBuffer; | 
					
						
							|  |  |  |     img->pBuffer.resize(img->pWidth * img->pHeight * 3); | 
					
						
							|  |  |  |     ImgConvert::RGB32toRGBA24(img->pBuffer, cpy, img->pWidth, img->pHeight); | 
					
						
							|  |  |  |     img->pFmt = RGB; | 
					
						
							| 
									
										
										
										
											2025-08-14 20:37:55 +02:00
										 |  |  |   } else if (img->pFmt == Image::RGBA && dst == Image::BGRA) { | 
					
						
							|  |  |  |     for (int i = 0; i < (img->pWidth * img->pHeight * 4); i += 4) { | 
					
						
							|  |  |  |       u8 _tmp = img->pBuffer[i + 0]; | 
					
						
							|  |  |  |       img->pBuffer[i + 0] = img->pBuffer[i + 2]; | 
					
						
							|  |  |  |       img->pBuffer[i + 2] = _tmp; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2025-06-22 21:05:09 +02:00
										 |  |  |   } else if (img->pFmt == Image::RGBA && dst == Image::RGB565) { | 
					
						
							|  |  |  |     Convert(img, Image::RGB); | 
					
						
							|  |  |  |     Convert(img, Image::RGB565); | 
					
						
							|  |  |  |   } else if (img->pFmt == Image::RGB && dst == Image::RGB565) { | 
					
						
							|  |  |  |     auto f = [](u8 r, u8 g, u8 b) -> u16 { | 
					
						
							|  |  |  |       u16 _r = (r >> 3); | 
					
						
							|  |  |  |       u16 _g = (g >> 2); | 
					
						
							|  |  |  |       u16 _b = (b >> 3); | 
					
						
							|  |  |  |       return (_r << 11) | (_g << 5) | _b; | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  |     std::vector<PD::u8> cpy = img->pBuffer; | 
					
						
							|  |  |  |     img->pBuffer.resize(img->pWidth * img->pHeight * 2); | 
					
						
							|  |  |  |     for (int y = 0; y < img->pWidth; y++) { | 
					
						
							|  |  |  |       for (int x = 0; x < img->pHeight; x++) { | 
					
						
							|  |  |  |         int src = (y * img->pWidth + x) * 3; | 
					
						
							|  |  |  |         int dst = (y * img->pWidth + x) * 2; | 
					
						
							|  |  |  |         u16 new_px = f(cpy[src + 0], cpy[src + 1], cpy[src + 2]); | 
					
						
							|  |  |  |         img->pBuffer[dst + 0] = new_px >> 8; | 
					
						
							|  |  |  |         img->pBuffer[dst + 1] = new_px & 0xff; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     img->pFmt = RGB565; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | PD_IMAGE_API int Image::Fmt2Bpp(Format fmt) { | 
					
						
							|  |  |  |   switch (fmt) { | 
					
						
							|  |  |  |     case RGBA: | 
					
						
							|  |  |  |     case ABGR: | 
					
						
							|  |  |  |       return 4; | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case RGB: | 
					
						
							|  |  |  |     case BGR: | 
					
						
							|  |  |  |       return 3; | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case RGB565: | 
					
						
							|  |  |  |       return 2; | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       return 0; | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | PD_IMAGE_API void Image::ReTile(Image::Ref img, | 
					
						
							|  |  |  |                                 std::function<u32(int x, int y, int w)> src, | 
					
						
							|  |  |  |                                 std::function<u32(int x, int y, int w)> dst) { | 
					
						
							|  |  |  |   std::vector<PD::u8> cpy = img->pBuffer; | 
					
						
							|  |  |  |   /** could use fmt here but for 565 that woulnt work as it is not supported by
 | 
					
						
							|  |  |  |    * file loading where fmt is used */ | 
					
						
							|  |  |  |   int bpp = Fmt2Bpp(img->pFmt); | 
					
						
							|  |  |  |   for (int y = 0; y < img->pHeight; y++) { | 
					
						
							|  |  |  |     for (int x = 0; x < img->pWidth; x++) { | 
					
						
							|  |  |  |       int src_idx = src(x, y, img->pWidth); | 
					
						
							|  |  |  |       int dst_idx = dst(x, y, img->pWidth); | 
					
						
							|  |  |  |       for (int i = 0; i < bpp; i++) { | 
					
						
							|  |  |  |         img->pBuffer[dst_idx + i] = cpy[src_idx + i]; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2025-03-05 12:06:40 +01:00
										 |  |  | }  // namespace PD
 |