123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113 |
- #include <list>
- #include <string>
- #include <vector>
- #include "png.h"
- #include "Common/FileUtil.h"
- #include "VideoCommon/ImageWrite.h"
- bool SaveData(const std::string& filename, const char* data)
- {
- std::ofstream f;
- OpenFStream(f, filename, std::ios::binary);
- f << data;
- return true;
- }
- bool TextureToPng(u8* data, int row_stride, const std::string& filename, int width, int height, bool saveAlpha)
- {
- bool success = false;
- if (!data)
- return false;
- char title[] = "Dolphin Screenshot";
- char title_key[] = "Title";
- png_structp png_ptr = nullptr;
- png_infop info_ptr = nullptr;
-
- File::IOFile fp(filename, "wb");
- if (!fp.IsOpen())
- {
- PanicAlertT("Screenshot failed: Could not open file \"%s\" (error %d)", filename.c_str(), errno);
- goto finalise;
- }
-
- png_ptr = png_create_write_struct(PNG_LIBPNG_VER_STRING, nullptr, nullptr, nullptr);
- if (png_ptr == nullptr)
- {
- PanicAlert("Screenshot failed: Could not allocate write struct");
- goto finalise;
- }
-
- info_ptr = png_create_info_struct(png_ptr);
- if (info_ptr == nullptr)
- {
- PanicAlert("Screenshot failed: Could not allocate info struct");
- goto finalise;
- }
-
- if (setjmp(png_jmpbuf(png_ptr)))
- {
- PanicAlert("Screenshot failed: Error during PNG creation");
- goto finalise;
- }
- png_init_io(png_ptr, fp.GetHandle());
-
- png_set_IHDR(png_ptr, info_ptr, width, height,
- 8, PNG_COLOR_TYPE_RGB_ALPHA, PNG_INTERLACE_NONE,
- PNG_COMPRESSION_TYPE_BASE, PNG_FILTER_TYPE_BASE);
- png_text title_text;
- title_text.compression = PNG_TEXT_COMPRESSION_NONE;
- title_text.key = title_key;
- title_text.text = title;
- png_set_text(png_ptr, info_ptr, &title_text, 1);
- png_write_info(png_ptr, info_ptr);
-
- for (auto y = 0; y < height; ++y)
- {
- u8* row_ptr = (u8*)data + y * row_stride;
- u8* ptr = row_ptr;
- for (auto x = 0; x < row_stride / 4; ++x)
- {
- if (!saveAlpha)
- ptr[3] = 0xff;
- ptr += 4;
- }
- png_write_row(png_ptr, row_ptr);
- }
-
- png_write_end(png_ptr, nullptr);
- success = true;
- finalise:
- if (info_ptr != nullptr) png_free_data(png_ptr, info_ptr, PNG_FREE_ALL, -1);
- if (png_ptr != nullptr) png_destroy_write_struct(&png_ptr, (png_infopp)nullptr);
- return success;
- }
|