#pragma once #include "EmberCommonPch.h" #define PNG_COMMENT_MAX 8 /// /// Write a PPM file. /// /// The full path and name of the file /// Pointer to the image data to write /// Width of the image in pixels /// Height of the image in pixels /// True if success, else false static bool WritePpm(const char* filename, byte* image, size_t width, size_t height) { bool b = false; size_t size = width * height * 3; FILE* file; if (fopen_s(&file, filename, "wb") == 0) { fprintf_s(file, "P6\n"); fprintf_s(file, "%lu %lu\n255\n", width, height); b = (size == fwrite(image, 1, size, file)); fclose(file); } return b; } /// /// Write a JPEG file. /// /// The full path and name of the file /// Pointer to the image data to write /// Width of the image in pixels /// Height of the image in pixels /// The quality to use /// True to embed comments, else false /// The comment string to embed /// Id of the author /// Url of the author /// Nickname of the author /// True if success, else false static bool WriteJpeg(const char* filename, byte* image, size_t width, size_t height, int quality, bool enableComments, EmberImageComments& comments, string id, string url, string nick) { bool b = false; FILE* file; if (fopen_s(&file, filename, "wb") == 0) { size_t i; jpeg_error_mgr jerr; jpeg_compress_struct info; char nickString[64], urlString[128], idString[128]; char bvString[64], niString[64], rtString[64]; char genomeString[65536], verString[64]; //Create the mandatory comment strings. snprintf_s(genomeString, 65536, "flam3_genome: %s", comments.m_Genome.c_str()); snprintf_s(bvString, 64, "flam3_error_rate: %s", comments.m_Badvals.c_str()); snprintf_s(niString, 64, "flam3_samples: %s", comments.m_NumIters.c_str()); snprintf_s(rtString, 64, "flam3_time: %s", comments.m_Runtime.c_str()); snprintf_s(verString, 64, "flam3_version: %s", EmberVersion()); info.err = jpeg_std_error(&jerr); jpeg_create_compress(&info); jpeg_stdio_dest(&info, file); info.in_color_space = JCS_RGB; info.input_components = 3; info.image_width = JDIMENSION(width); info.image_height = JDIMENSION(height); jpeg_set_defaults(&info); jpeg_set_quality(&info, quality, TRUE); jpeg_start_compress(&info, TRUE); //Write comments to jpeg. if (enableComments) { jpeg_write_marker(&info, JPEG_COM, reinterpret_cast(verString), uint(strlen(verString))); if (nick != "") { snprintf_s(nickString, 64, "flam3_nickname: %s", nick.c_str()); jpeg_write_marker(&info, JPEG_COM, reinterpret_cast(nickString), uint(strlen(nickString))); } if (url != "") { snprintf_s(urlString, 128, "flam3_url: %s", url.c_str()); jpeg_write_marker(&info, JPEG_COM, reinterpret_cast(urlString), uint(strlen(urlString))); } if (id != "") { snprintf_s(idString, 128, "flam3_id: %s", id.c_str()); jpeg_write_marker(&info, JPEG_COM, reinterpret_cast(idString), uint(strlen(idString))); } jpeg_write_marker(&info, JPEG_COM, reinterpret_cast(bvString), uint(strlen(bvString))); jpeg_write_marker(&info, JPEG_COM, reinterpret_cast(niString), uint(strlen(niString))); jpeg_write_marker(&info, JPEG_COM, reinterpret_cast(rtString), uint(strlen(rtString))); jpeg_write_marker(&info, JPEG_COM, reinterpret_cast(genomeString), uint(strlen(genomeString))); } for (i = 0; i < height; i++) { JSAMPROW row_pointer[1]; row_pointer[0] = reinterpret_cast(image) + (3 * width * i); jpeg_write_scanlines(&info, row_pointer, 1); } jpeg_finish_compress(&info); jpeg_destroy_compress(&info); fclose(file); b = true; } return b; } /// /// Write a PNG file. /// /// The full path and name of the file /// Pointer to the image data to write /// Width of the image in pixels /// Height of the image in pixels /// Bytes per channel, 1 or 2. /// True to embed comments, else false /// The comment string to embed /// Id of the author /// Url of the author /// Nickname of the author /// True if success, else false static bool WritePng(const char* filename, byte* image, size_t width, size_t height, size_t bytesPerChannel, bool enableComments, EmberImageComments& comments, string id, string url, string nick) { bool b = false; FILE* file; if (fopen_s(&file, filename, "wb") == 0) { png_structp png_ptr; png_infop info_ptr; png_text text[PNG_COMMENT_MAX]; size_t i; glm::uint16 testbe = 1; vector rows(height); text[0].compression = PNG_TEXT_COMPRESSION_NONE; text[0].key = const_cast("flam3_version"); text[0].text = const_cast(EmberVersion()); text[1].compression = PNG_TEXT_COMPRESSION_NONE; text[1].key = const_cast("flam3_nickname"); text[1].text = const_cast(nick.c_str()); text[2].compression = PNG_TEXT_COMPRESSION_NONE; text[2].key = const_cast("flam3_url"); text[2].text = const_cast(url.c_str()); text[3].compression = PNG_TEXT_COMPRESSION_NONE; text[3].key = const_cast("flam3_id"); text[3].text = const_cast(id.c_str()); text[4].compression = PNG_TEXT_COMPRESSION_NONE; text[4].key = const_cast("flam3_error_rate"); text[4].text = const_cast(comments.m_Badvals.c_str()); text[5].compression = PNG_TEXT_COMPRESSION_NONE; text[5].key = const_cast("flam3_samples"); text[5].text = const_cast(comments.m_NumIters.c_str()); text[6].compression = PNG_TEXT_COMPRESSION_NONE; text[6].key = const_cast("flam3_time"); text[6].text = const_cast(comments.m_Runtime.c_str()); text[7].compression = PNG_TEXT_COMPRESSION_zTXt; text[7].key = const_cast("flam3_genome"); text[7].text = const_cast(comments.m_Genome.c_str()); for (i = 0; i < height; i++) rows[i] = image + i * width * 4 * bytesPerChannel; png_ptr = png_create_write_struct(PNG_LIBPNG_VER_STRING, nullptr, nullptr, nullptr); info_ptr = png_create_info_struct(png_ptr); if (setjmp(png_jmpbuf(png_ptr))) { fclose(file); png_destroy_write_struct(&png_ptr, &info_ptr); perror("writing file"); return false; } png_init_io(png_ptr, file); png_set_IHDR(png_ptr, info_ptr, png_uint_32(width), png_uint_32(height), 8 * png_uint_32(bytesPerChannel), PNG_COLOR_TYPE_RGBA, PNG_INTERLACE_NONE, PNG_COMPRESSION_TYPE_BASE, PNG_FILTER_TYPE_BASE); if (enableComments == 1) png_set_text(png_ptr, info_ptr, text, PNG_COMMENT_MAX); png_write_info(png_ptr, info_ptr); //Must set this after png_write_info(). if (bytesPerChannel == 2 && testbe != htons(testbe)) { png_set_swap(png_ptr); } png_write_image(png_ptr, const_cast(rows.data())); png_write_end(png_ptr, info_ptr); png_destroy_write_struct(&png_ptr, &info_ptr); fclose(file); b = true; } return b; } /// /// Convert an RGB buffer to BGR for usage with BMP. /// /// The buffer to convert /// The width. /// The height. /// The size of the new buffer created /// The converted buffer if successful, else NULL. static byte* ConvertRGBToBMPBuffer(byte* buffer, size_t width, size_t height, size_t& newSize) { if (buffer == nullptr || width == 0 || height == 0) return nullptr; size_t padding = 0; size_t scanlinebytes = width * 3; while ((scanlinebytes + padding ) % 4 != 0) padding++; size_t psw = scanlinebytes + padding; newSize = height * psw; byte* newBuf = new byte[newSize]; if (newBuf) { memset (newBuf, 0, newSize); size_t bufpos = 0; size_t newpos = 0; for (size_t y = 0; y < height; y++) { for (size_t x = 0; x < 3 * width; x += 3) { bufpos = y * 3 * width + x; // position in original buffer newpos = (height - y - 1) * psw + x; // position in padded buffer newBuf[newpos] = buffer[bufpos+2]; // swap r and b newBuf[newpos + 1] = buffer[bufpos + 1]; // g stays newBuf[newpos + 2] = buffer[bufpos]; // swap b and r //No swap. //newBuf[newpos] = buffer[bufpos]; //newBuf[newpos + 1] = buffer[bufpos + 1]; //newBuf[newpos + 2] = buffer[bufpos + 2]; } } return newBuf; } return nullptr; } /// /// Save a Bmp file. /// /// The full path and name of the file /// Pointer to the image data to write /// Width of the image in pixels /// Height of the image in pixels /// Padded size, greater than or equal to total image size. /// True if success, else false static bool SaveBmp(const char* filename, byte* image, size_t width, size_t height, size_t paddedSize) { #ifdef WIN32 BITMAPFILEHEADER bmfh; BITMAPINFOHEADER info; DWORD bwritten; HANDLE file; memset (&bmfh, 0, sizeof (BITMAPFILEHEADER)); memset (&info, 0, sizeof (BITMAPINFOHEADER)); bmfh.bfType = 0x4d42; // 0x4d42 = 'BM' bmfh.bfReserved1 = 0; bmfh.bfReserved2 = 0; bmfh.bfSize = sizeof(BITMAPFILEHEADER) + sizeof(BITMAPINFOHEADER) + (DWORD)paddedSize; bmfh.bfOffBits = 0x36; info.biSize = sizeof(BITMAPINFOHEADER); info.biWidth = (LONG)width; info.biHeight = (LONG)height; info.biPlanes = 1; info.biBitCount = 24; info.biCompression = BI_RGB; info.biSizeImage = 0; info.biXPelsPerMeter = 0x0ec4; info.biYPelsPerMeter = 0x0ec4; info.biClrUsed = 0; info.biClrImportant = 0; if ((file = CreateFileA(filename, GENERIC_WRITE, FILE_SHARE_READ, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL)) == NULL) { CloseHandle(file); return false; } if (WriteFile(file, &bmfh, sizeof (BITMAPFILEHEADER), &bwritten, NULL) == false) { CloseHandle(file); return false; } if (WriteFile(file, &info, sizeof(BITMAPINFOHEADER), &bwritten, NULL) == false) { CloseHandle(file); return false; } if (WriteFile(file, image, (DWORD)paddedSize, &bwritten, NULL) == false) { CloseHandle(file); return false; } CloseHandle(file); #endif return true; } /// /// Convert a buffer from RGB to BGR and write a Bmp file. /// /// The full path and name of the file /// Pointer to the image data to write /// Width of the image in pixels /// Height of the image in pixels /// True if success, else false static bool WriteBmp(const char* filename, byte* image, size_t width, size_t height) { bool b = false; size_t newSize; unique_ptr bgrBuf(ConvertRGBToBMPBuffer(image, width, height, newSize)); if (bgrBuf.get()) b = SaveBmp(filename, bgrBuf.get(), width, height, newSize); return b; }