aboutsummaryrefslogtreecommitdiff
path: root/ImageManipulation.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'ImageManipulation.cpp')
-rw-r--r--ImageManipulation.cpp208
1 files changed, 206 insertions, 2 deletions
diff --git a/ImageManipulation.cpp b/ImageManipulation.cpp
index 3017908..51c0601 100644
--- a/ImageManipulation.cpp
+++ b/ImageManipulation.cpp
@@ -6,9 +6,195 @@ ImageManipulation::ImageManipulation() {}
ImageManipulation::~ImageManipulation() {}
-void ImageManipulation::SetJpegFromBuffer(
+struct png_memory_io {
+ std::vector<unsigned char> &image_buffer;
+ size_t offset;
+};
+
+static void read_png_buffer_from_memory(png_structp png_ptr, png_bytep outBytes,
+ png_size_t byteCountToRead) {
+ png_voidp io_ptr = png_get_io_ptr(png_ptr);
+ if (io_ptr == NULL) {
+ std::cerr << "Read PNG buffer from memory failed: IO Pointer NULL"
+ << std::endl;
+ return;
+ }
+
+ struct png_memory_io *pmi = (struct png_memory_io *)io_ptr;
+ size_t maxBytesToRead =
+ (pmi->image_buffer.size() - pmi->offset > byteCountToRead
+ ? byteCountToRead
+ : pmi->image_buffer.size() - pmi->offset);
+ memcpy(outBytes, pmi->image_buffer.data() + pmi->offset, maxBytesToRead);
+ pmi->offset += maxBytesToRead;
+}
+
+// More or less copy-pasta from CImg.h's _load_png(...)
+static bool load_png_buffer(std::vector<unsigned char> &image_buffer,
+ cimg_library::CImg<unsigned char> &image,
+ unsigned int *const bits_per_value) {
+ png_structp png_ptr;
+ png_infop info_ptr, end_info;
+ struct png_memory_io pmi = {.image_buffer = image_buffer, .offset = 0};
+
+ png_uint_32 W, H;
+ int bit_depth, color_type, interlace_type;
+ bool is_gray = false;
+
+ png_ptr = png_create_read_struct(PNG_LIBPNG_VER_STRING, NULL, NULL, NULL);
+ if (png_ptr == NULL) {
+ return false;
+ }
+
+ info_ptr = png_create_info_struct(png_ptr);
+ if (info_ptr == NULL) {
+ png_destroy_read_struct(&png_ptr, NULL, NULL);
+ return false;
+ }
+
+ end_info = png_create_info_struct(png_ptr);
+ if (end_info == NULL) {
+ png_destroy_read_struct(&png_ptr, &info_ptr, NULL);
+ return false;
+ }
+
+ png_set_read_fn(png_ptr, &pmi, read_png_buffer_from_memory);
+ png_set_sig_bytes(png_ptr, 0);
+ png_read_info(png_ptr, info_ptr);
+
+ png_get_IHDR(png_ptr, info_ptr, &W, &H, &bit_depth, &color_type,
+ &interlace_type, NULL, NULL);
+ png_set_interlace_handling(png_ptr);
+ if (bits_per_value != NULL) {
+ *bits_per_value = (unsigned int)bit_depth;
+ }
+
+ if (color_type == PNG_COLOR_TYPE_PALETTE) {
+ png_set_palette_to_rgb(png_ptr);
+ color_type = PNG_COLOR_TYPE_RGB;
+ bit_depth = 8;
+ }
+ if (color_type == PNG_COLOR_TYPE_GRAY && bit_depth < 8) {
+ png_set_expand_gray_1_2_4_to_8(png_ptr);
+ is_gray = true;
+ bit_depth = 8;
+ }
+ if (png_get_valid(png_ptr, info_ptr, PNG_INFO_tRNS) != 0) {
+ png_set_tRNS_to_alpha(png_ptr);
+ color_type |= PNG_COLOR_MASK_ALPHA;
+ }
+ if (color_type == PNG_COLOR_TYPE_GRAY ||
+ color_type == PNG_COLOR_TYPE_GRAY_ALPHA) {
+ png_set_gray_to_rgb(png_ptr);
+ color_type |= PNG_COLOR_MASK_COLOR;
+ is_gray = true;
+ }
+ if (color_type == PNG_COLOR_TYPE_RGB) {
+ png_set_filler(png_ptr, 0xffffU, PNG_FILLER_AFTER);
+ }
+
+ png_read_update_info(png_ptr, info_ptr);
+ if (bit_depth != 8 && bit_depth != 16) {
+ png_destroy_read_struct(&png_ptr, &end_info, NULL);
+ return false;
+ }
+ const int byte_depth = bit_depth >> 3;
+
+ png_bytep *const imgData = new png_bytep[H];
+ for (unsigned int row = 0; row < H; ++row) {
+ imgData[row] = new png_byte[(size_t)byte_depth * 4 * W];
+ }
+ png_read_image(png_ptr, imgData);
+ png_read_end(png_ptr, end_info);
+
+ if (color_type != PNG_COLOR_TYPE_RGB &&
+ color_type != PNG_COLOR_TYPE_RGB_ALPHA) {
+ png_destroy_read_struct(&png_ptr, &end_info, (png_infopp)0);
+ return false;
+ }
+ const bool is_alpha = (color_type == PNG_COLOR_TYPE_RGBA);
+ try {
+ image.assign(W, H, 1,
+ (is_gray == true ? 1 : 3) + (is_alpha == true ? 1 : 0));
+ } catch (...) {
+ throw;
+ }
+
+ unsigned char *ptr_r = image.data(0, 0, 0, 0);
+ unsigned char *ptr_g = (is_gray == true ? 0 : image.data(0, 0, 0, 1));
+ unsigned char *ptr_b = (is_gray == true ? 0 : image.data(0, 0, 0, 2));
+ unsigned char *ptr_a =
+ (is_alpha == false ? 0 : image.data(0, 0, 0, is_gray == true ? 1 : 3));
+
+ switch (bit_depth) {
+ case 8: {
+ cimg_forY(image, y) {
+ const unsigned char *ptrs = imgData[y];
+ cimg_forX(image, x) {
+ *(ptr_r++) = *(ptrs++);
+ if (ptr_g)
+ *(ptr_g++) = *(ptrs++);
+ else
+ ++ptrs;
+ if (ptr_b)
+ *(ptr_b++) = *(ptrs++);
+ else
+ ++ptrs;
+ if (ptr_a)
+ *(ptr_a++) = *(ptrs++);
+ else
+ ++ptrs;
+ }
+ }
+ } break;
+ case 16: {
+ cimg_forY(image, y) {
+ unsigned short const *ptrs = (unsigned short *)imgData[y];
+ if (cimg_library::cimg::endianness() == false) {
+ cimg_library::cimg::invert_endianness(ptrs, 4 * image._width);
+ }
+ cimg_forX(image, x) {
+ *(ptr_r++) = *(ptrs++);
+ if (ptr_g)
+ *(ptr_g++) = *(ptrs++);
+ else
+ ++ptrs;
+ if (ptr_b)
+ *(ptr_b++) = *(ptrs++);
+ else
+ ++ptrs;
+ if (ptr_a)
+ *(ptr_a++) = *(ptrs++);
+ else
+ ++ptrs;
+ }
+ }
+ } break;
+ }
+
+ png_destroy_read_struct(&png_ptr, &info_ptr, &end_info);
+
+ cimg_forY(image, n) { delete[] imgData[n]; }
+ delete[] imgData;
+
+ return true;
+}
+
+bool ImageManipulation::SetImageFromBuffer(
std::vector<unsigned char> &image_buffer) {
- image.load_jpeg_buffer(image_buffer.data(), image_buffer.size());
+ switch (GetImageFormat(image_buffer)) {
+ case IF_UNSUPPORTED:
+ goto failure;
+ case IF_JPEG:
+ image.load_jpeg_buffer(image_buffer.data(), image_buffer.size());
+ return true;
+ case IF_PNG:
+ return load_png_buffer(image_buffer, image, NULL);
+ }
+
+failure:
+ std::cerr << "Unsupported Image Format.." << std::endl;
+ return false;
}
void ImageManipulation::SaveToFile(std::string filename) {
@@ -31,3 +217,21 @@ void ImageManipulation::Ukrainify(float opacity) {
image.draw_rectangle(0, image.height() / 2, image.width(), image.height(),
lower_color, opacity);
}
+
+enum ImageFormat
+ImageManipulation::GetImageFormat(std::vector<unsigned char> &image_buffer) {
+ if (image_buffer.size() < 8) {
+ return IF_UNSUPPORTED;
+ }
+
+ if (image_buffer[0] == 0xFF && image_buffer[1] == 0xD8 &&
+ image_buffer[2] == 0xFF) {
+ return IF_JPEG;
+ }
+
+ if (png_sig_cmp(image_buffer.data(), 0, 8) == 0) {
+ return IF_PNG;
+ }
+
+ return IF_UNSUPPORTED;
+}