2013-04-18 05:09:55 +02:00
|
|
|
// Copyright 2013 Dolphin Emulator Project
|
|
|
|
// Licensed under GPLv2
|
|
|
|
// Refer to the license.txt file included.
|
2008-12-08 06:25:12 +01:00
|
|
|
|
|
|
|
#include <list>
|
|
|
|
#include <vector>
|
|
|
|
|
2013-11-15 01:00:38 +01:00
|
|
|
#include "png.h"
|
2008-12-08 06:25:12 +01:00
|
|
|
#include "ImageWrite.h"
|
2011-03-11 11:21:46 +01:00
|
|
|
#include "FileUtil.h"
|
2008-12-08 06:25:12 +01:00
|
|
|
|
|
|
|
#pragma pack(push, 1)
|
|
|
|
|
|
|
|
struct TGA_HEADER
|
|
|
|
{
|
2013-03-20 02:51:12 +01:00
|
|
|
u8 identsize; // size of ID field that follows 18 u8 header (0 usually)
|
|
|
|
u8 colourmaptype; // type of colour map 0=none, 1=has palette
|
|
|
|
u8 imagetype; // type of image 0=none,1=indexed,2=rgb,3=grey,+8=rle packed
|
|
|
|
|
|
|
|
s16 colourmapstart; // first colour map entry in palette
|
|
|
|
s16 colourmaplength; // number of colours in palette
|
|
|
|
u8 colourmapbits; // number of bits per palette entry 15,16,24,32
|
|
|
|
|
|
|
|
s16 xstart; // image x origin
|
|
|
|
s16 ystart; // image y origin
|
|
|
|
s16 width; // image width in pixels
|
|
|
|
s16 height; // image height in pixels
|
|
|
|
u8 bits; // image bits per pixel 8,16,24,32
|
|
|
|
u8 descriptor; // image descriptor bits (vh flip bits)
|
2013-10-29 06:23:17 +01:00
|
|
|
|
2013-03-20 02:51:12 +01:00
|
|
|
// pixel data follows header
|
2008-12-08 06:25:12 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
#pragma pack(pop)
|
|
|
|
|
|
|
|
bool SaveTGA(const char* filename, int width, int height, void* pdata)
|
|
|
|
{
|
2011-03-11 11:21:46 +01:00
|
|
|
TGA_HEADER hdr;
|
|
|
|
File::IOFile f(filename, "wb");
|
|
|
|
if (!f)
|
|
|
|
return false;
|
2008-12-08 06:25:12 +01:00
|
|
|
|
2011-03-11 11:21:46 +01:00
|
|
|
_assert_(sizeof(TGA_HEADER) == 18 && sizeof(hdr) == 18);
|
2013-03-20 02:51:12 +01:00
|
|
|
|
2011-03-11 11:21:46 +01:00
|
|
|
memset(&hdr, 0, sizeof(hdr));
|
|
|
|
hdr.imagetype = 2;
|
|
|
|
hdr.bits = 32;
|
|
|
|
hdr.width = width;
|
|
|
|
hdr.height = height;
|
|
|
|
hdr.descriptor |= 8|(1<<5); // 8bit alpha, flip vertical
|
|
|
|
|
|
|
|
f.WriteArray(&hdr, 1);
|
|
|
|
f.WriteBytes(pdata, width * height * 4);
|
|
|
|
|
|
|
|
return true;
|
2008-12-08 06:25:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
bool SaveData(const char* filename, const char* data)
|
|
|
|
{
|
2013-03-01 02:33:39 +01:00
|
|
|
std::ofstream f;
|
|
|
|
OpenFStream(f, filename, std::ios::binary);
|
2011-03-11 11:21:46 +01:00
|
|
|
f << data;
|
|
|
|
|
|
|
|
return true;
|
2008-12-08 06:25:12 +01:00
|
|
|
}
|
2013-11-15 01:00:38 +01:00
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
TextureToPng
|
|
|
|
|
|
|
|
Inputs:
|
|
|
|
data : This is an array of RGBA with 8 bits per channel. 4 bytes for each pixel.
|
|
|
|
data is a newly allocated memory and must have delete[] run on it before returning.
|
|
|
|
|
|
|
|
row_stride: Determines the amount of bytes per row of pixels.
|
|
|
|
*/
|
|
|
|
bool TextureToPng(u8* data, int row_stride, const char* filename, int width, int height, bool saveAlpha)
|
|
|
|
{
|
|
|
|
bool success = false;
|
|
|
|
|
|
|
|
if (!data)
|
|
|
|
return false;
|
|
|
|
|
2013-11-15 01:05:03 +01:00
|
|
|
char title[] = "Dolphin Screenshot";
|
|
|
|
FILE *fp = NULL;
|
|
|
|
png_structp png_ptr = NULL;
|
|
|
|
png_infop info_ptr = NULL;
|
|
|
|
|
2013-11-15 01:00:38 +01:00
|
|
|
// Open file for writing (binary mode)
|
2013-11-15 01:05:03 +01:00
|
|
|
fp = fopen(filename, "wb");
|
2013-11-15 01:00:38 +01:00
|
|
|
if (fp == NULL) {
|
|
|
|
PanicAlert("Screenshot failed: Could not open file %s\n", filename);
|
|
|
|
goto finalise;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Initialize write structure
|
2013-11-15 01:05:03 +01:00
|
|
|
png_ptr = png_create_write_struct(PNG_LIBPNG_VER_STRING, NULL, NULL, NULL);
|
2013-11-15 01:00:38 +01:00
|
|
|
if (png_ptr == NULL) {
|
|
|
|
PanicAlert("Screenshot failed: Could not allocate write struct\n");
|
|
|
|
goto finalise;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
// Initialize info structure
|
2013-11-15 01:05:03 +01:00
|
|
|
info_ptr = png_create_info_struct(png_ptr);
|
2013-11-15 01:00:38 +01:00
|
|
|
if (info_ptr == NULL) {
|
|
|
|
PanicAlert("Screenshot failed: Could not allocate info struct\n");
|
|
|
|
goto finalise;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Setup Exception handling
|
|
|
|
if (setjmp(png_jmpbuf(png_ptr))) {
|
|
|
|
PanicAlert("Screenshot failed: Error during png creation\n");
|
|
|
|
goto finalise;
|
|
|
|
}
|
|
|
|
|
|
|
|
png_init_io(png_ptr, fp);
|
|
|
|
|
|
|
|
// Write header (8 bit colour depth)
|
|
|
|
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";
|
|
|
|
title_text.text = title;
|
|
|
|
png_set_text(png_ptr, info_ptr, &title_text, 1);
|
|
|
|
|
|
|
|
png_write_info(png_ptr, info_ptr);
|
|
|
|
|
|
|
|
// Write image data
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
|
|
|
// End write
|
|
|
|
png_write_end(png_ptr, NULL);
|
|
|
|
|
|
|
|
success = true;
|
|
|
|
|
|
|
|
finalise:
|
|
|
|
|
|
|
|
if (fp != NULL) fclose(fp);
|
|
|
|
if (info_ptr != NULL) png_free_data(png_ptr, info_ptr, PNG_FREE_ALL, -1);
|
|
|
|
if (png_ptr != NULL) png_destroy_write_struct(&png_ptr, (png_infopp)NULL);
|
|
|
|
|
|
|
|
// Our duty to delete the inputted data.
|
|
|
|
delete[] data;
|
|
|
|
|
|
|
|
return success;
|
|
|
|
}
|